Java细粒度锁实现的3种方式 最近在工作上碰见了一些高并发的场景需要加锁来保证业务逻辑的正确性,并且要求加锁后性能不能受到太大的影响。...而java自身api提供的锁粒度太大,很难同时满足这些需求,于是自己动手写了几个简单的扩展… ? 1....哈希锁 上述分段锁的基础上发展起来的第二种锁策略,目的是实现真正意义上的细粒度锁。每个哈希值不同的对象都能获得自己独立的锁。在测试中,在被锁住的代码执行速度飞快的情况下,效率比分段锁慢 30% 左右。...弱引用锁 哈希锁因为引入的分段锁来保证锁创建和销毁的同步,总感觉有点瑕疵,所以写了第三个锁来寻求更好的性能和更细粒度的锁。...,写着写着发现正在实现的东西和java 原生的锁区别不大,于是放弃改为对java自带锁的封装,浪费了不少时间。
java中的乐观锁基本都是通过CAS操作实现的,CAS是一种更新的原子操作,比较当前值跟传入值是否一样,一样则更新,否则失败。...java中的悲观锁就是Synchronized,AQS框架下的锁则是先尝试cas乐观锁去获取锁,获取不到,才会转换为悲观锁,如RetreenLock。...java中的锁 1.自旋锁 自旋锁原理非常简单,如果持有锁的线程能在很短时间内释放锁资源,那么那些等待竞争锁的线程就不需要做内核态和用户态之间的切换进入阻塞挂起状态,它们只需要等一等(自旋),等持有锁的线程释放锁后即可立即获取锁...synchronized会导致争用不到锁的线程进入阻塞状态,所以说它是java语言中一个重量级的同步操纵,被称为重量级锁,为了缓解上述性能问题,JVM从1.5开始,引入了轻量锁与偏向锁,默认启用了自旋锁...偏向锁 Java偏向锁(Biased Locking)是Java6引入的一项多线程优化。
java 中的乐观锁基本都是通过 CAS 操作实现的,CAS 是一种更新的原子操作,比较当前值跟传入 值是否一样,一样则更新,否则失败。...java中的悲观锁就是Synchronized,AQS框架下的锁则是先尝试cas乐观锁去获取锁,获取不到, 才会转换为悲观锁,如 RetreenLock。...3.自旋锁 自旋锁原理非常简单,如果持有锁的线程能在很短时间内释放锁资源,那么那些等待竞争锁 的线程就不需要做内核态和用户态之间的切换进入阻塞挂起状态,它们只需要等一等(自旋), 等持有锁的线程释放锁后即可立即获取锁...但是如果锁的竞争激烈,或者持有锁的线程需要长时间占用锁执行同步块,这时候就不适合 使用自旋锁了,因为自旋锁在获取锁前一直都是占用 cpu 做无用功,占着 XX 不 XX,同时有大量 线程在竞争一个锁,会导致获取锁的时间很长...所以这种情况下我们要关闭自旋锁;自旋锁时间阈值(1.6 引入了适应性自旋锁) 自旋锁的目的是为了占着 CPU 的资源不释放,等到获取到锁立即进行处理。但是如何去选择 自旋的执行时间呢?
悲观锁 & 乐观锁悲观锁:认为多个线程访问同一个共享变量冲突的概率较大,会在每次访问共享变量之前都去真正加锁。Java中的 synchronized 锁 和 ReentrantLock就是悲观锁。...Java中的CAS就是乐观锁,比较当前值(主内存中的值)与预期值(当前线程中的值,主内存中值的一份拷贝)是否一样,一样则更新,否则继续进行CAS操作。...读写锁Java 标准库提供了 ReentrantReadWriteLock 类, 实现了读写锁。...自旋锁如果获取锁失败,立即再尝试获取锁,无限循环,直到获取到锁为止。第一次获取锁失败,第二次的尝试会在极短的时间内到来,一旦锁被其他线程释放,就能第一时间获取到锁。...同步锁同步锁表示并发执行的多个线程,在同一时间内只允许一个线程访问共享数据。在java中 synchronized 锁就是同步锁。
可重入锁递归锁,同一个线程,外层函数获得锁,内层的也获得锁。...finally { lock.unlock(); } }}多线程的判断用while if只适合两个线程的判断synchronized和lock的区别参考文章存在层次上synchronized: Java...的关键字,在jvm层面上Lock: 是一个接口锁的释放synchronized: 1、以获取锁的线程执行完同步代码,释放锁 2、线程执行发生异常,jvm会让线程释放锁Lock: 在finally中必须释放锁...: 在发生异常时候会自动释放占有的锁,因此不会出现死锁Lock: 发生异常时候,不会主动释放占有的锁,必须手动unlock来释放锁,可能引起死锁的发生锁的状态synchronized: 无法判断Lock...当线程执行遇到monitorenter指令时会尝试获取内置锁,如果获取锁则锁计数器+1,如果没有获取锁则阻塞;当遇到monitorexit指令时锁计数器-1,如果计数器为0则释放锁。
一、 Java锁 1.常见的锁有synchronized和Lock() ①synchronized 是jvm层面实现的,可以直接用,不过要锁住某个对象;lock是属于j.u.c包下的接口,用的时候要实现...@lock锁与synchronized相比,lock锁添加一些其他特性,如中断锁等候和定时锁等候。...2.悲观锁与乐观锁 ①悲观锁认为世界是悲观的,当去拿数据的时候就上锁,这样别人想拿这个锁就会阻塞直到拿到锁,传统的数据库用到了这种锁,像行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。...再比如Java里面的同步原语synchronized关键字的实现也是悲观锁。 ②乐观锁,认为一般并发是不会发生的,所以不会上锁。...基于CAS(无锁编程)实现,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制(解决ABA问题)。乐观锁适用于多读的应用类型,这样可以提高吞吐量。
Java中导致饥饿的原因: 高优先级线程吞噬所有的低优先级线程的CPU时间。 线程被永久堵塞在一个等待进入同步块的状态,因为其他线程总是能在它之前持续地对该同步块进行访问。 ...传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。再比如Java里面的同步原语synchronized关键字的实现也是悲观锁。...在Java中java.util.concurrent.atomic包下面的原子变量类就是使用了乐观锁的一种实现方式CAS实现的。...java中的Compare and Swap即CAS ,当多个线程尝试使用CAS同时更新同一个变量时,只有其中一个线程能更新变量的值,而其它线程都失败,失败的线程并不会被挂起,而是被告知这次竞争中失败,...从Java1.5开始JDK的atomic包里提供了一个类AtomicStampedReference来解决ABA问题。
synchronized关键字,我们一般称之为“同步锁”,用它来修饰需要同步的方法和需要同步代码块,默认是当前对象作为锁的对象。...在用类修饰synchronized时(或者修饰静态方法),默认是当前类的Class对象作为锁的对象,故存在着方法锁、对象锁、类锁这样的概念。 先给出以下代码感受下代码执行的时候为什么需要同步?...其他线程打不开这个锁,只能在外边等该线程释放掉该锁,一般都都是执行玩所有代码逻辑主动释放锁),表示此方法是当前线程独占的,对应到上述业务中就是一次只能有一个队列报数。...numberOff(); }, "队列" + i).start(); } } } 代码执行结果如下: 最后用线程池将上述代码写一下: package ioo; import java.util.concurrent.ExecutorService...; import java.util.concurrent.Executors; class SynchronizedExample { protected static int num =
我们都知道在 Java 中为了保证一些操作的安全性,就会涉及到使用锁,但是你对 Java 的锁了解的有多少呢?Java 都有哪些锁?以及他们是怎么实现的,今天了不起就来说说关于 Java 的锁。...在 Java 中,乐观锁通常是通过版本号、时间戳或其他状态信息来实现的。以下是乐观锁在 Java 中的一些常见实现方式: 版本号机制: 数据表中增加一个“版本号”字段。 读取数据时,同时读取版本号。...ReentrantLock类: Java的java.util.concurrent.locks.ReentrantLock类提供了重入锁的实现,这是一种悲观锁。...递归锁 Java中的递归锁(ReentrantLock)是java.util.concurrent.locks包下提供的一种可重入的互斥锁,它是悲观锁的一种实现。...读写锁 Java中的读写锁(ReadWriteLock)是一种允许多个读线程和单个写线程访问共享资源的同步机制。
参考链接: 了解Java中的类和对象 前言 之前看到过一篇文章,写的就是Java关键字synchronized的类锁和对象锁,今天想重温一下无奈发现文章已经搜索不到,百度之大部分都是重复的那么几篇文章...于是重写一篇博客介绍 synchronized 类锁 对象锁。 Java原生提供了 synchronized 关键字用于多线程编程,但往往入门使用者在发现使用情况与预期有差别,可阅读此文章。 ...Java的 synchronized 锁的是对象,也只锁对象: 对象锁是基于对堆内存内对象的头部加锁信息; 类锁是基于对类对应的 java.lang.Class对象加锁信息; 特别的, synchronized...例如 ReentrantLock是基于Java关键字volatile和CPU的CAS机制来实现的,若有知晓可在留言区告知一二 *) 类锁原理及为何类锁完全互斥 可优先阅读该篇文章图解Java类加载机制... 想获得一个Java的对象,则需要先获得Java的一个类,这便是Java的类加载。
由于具体业务场景的需求,需要保证数据在分布式环境下的正确更新,所以研究了一下Java中分布式锁的实现。...Java分布式锁的实现方式主要有以下三种: 数据库实现的乐观锁 Redis实现的分布式锁 Zookeeper实现的分布式锁 其中,较常用的是前两种方式,但是数据库实现方式需要较多的数据库操作,所以最终选择的是用...第一,Redis锁需要有一个超时时间,这样即便某个持有锁的节点挂了,也不到导致其他节点死锁,保证每个锁有一个UniqueId;第二,每个锁需要有一个UniqueId,确保当一个线程执行完一个任务去释放锁的时候释放的一定是自己的锁...,否则可能存在一种场景,就是一个线程释放锁的时候,它的锁可能已经超时被释放了,而因为缺少一个UniqueId,它却释放了另一个线程的锁 基于以上两点的考虑,分别设计了获取锁和释放锁的api。...若做check的时候,检查得到确实是这个锁的UniqueId,但是在执行del方法之前,这个锁已经超时,然后新的线程也已经获取到锁了,那么del删掉的锁,便不是自己的锁,而是下一个线程的锁。
java中的悲观锁就是Synchronized,AQS框架下的锁则是先尝试cas乐观锁去获取锁,获取不到,才会转换为悲观锁,如RetreenLock。...明确java线程切换的代价,是理解java中各种锁的优缺点的基础之一。...基础知识之三:markword 在介绍java锁之前,先说下什么是markword,markword是java对象数据结构中的一部分,要详细了解java对象的结构可以点击这里,这里只做markword的详细介绍...了解了markword结构,有助于后面了解java锁的加锁解锁过程; 小结 前面提到了java的4种锁,他们分别是重量级锁、自旋锁、轻量级锁和偏向锁, 不同的锁有不同特点,每种锁只有在其特定的场景下...偏向锁 Java偏向锁(Biased Locking)是Java6引入的一项多线程优化。
公平锁:是指多个线程按照申请锁的顺序来获取锁,有点先来后到的意思。...非公平锁:指多个线程获取锁的顺序并不是按照申请锁的顺序,上来就尝试占有锁,如果尝试失败,就再采用类似公平锁的方式获取锁。...对于Synchronized 锁也是一种非公平锁。 可重入锁(又名递归锁):指同一线程外层函数获得锁之后,内层递归函数仍然能获取该锁的代码。...(写锁):指该锁只能被一个线程所持有。...对 ReentrantLock 和 Synchronized 而言都是独占锁。 【共享锁】(读锁):指该锁可被多个线程持有。
Java中的分很多种类,按照场景的不同、特性的不同等分为了很多类,下面就来讲讲Java中锁的概念: 自旋锁:是指当一个线程在获取锁的时候,该锁已经被其他线程占用,则该线程会循环等待,并会不断检查是否成功获取锁...Java中悲观锁和乐观锁的实现 在Java语言中,对于悲观锁和乐观锁有不同的实现。 synchronized关键字和Lock相关实现类都是悲观锁。...只要一个线程获取到了对象的锁,会修改对象头中的Mark Word状态,同时线程中也会保存对应的状态。 而Java中的乐观锁最常采用的是CAS算法。...悲观锁和乐观锁的应用场景 悲观锁适用于写多读少的场景,操作资源的时候先加锁可以保证资源的正确性 乐观锁适用于读多写少的场景, 不加锁可以让读取数据的效率大幅增强 自旋锁和适应性自旋锁 阻塞或唤醒一个Java...无锁、偏向锁、轻量级锁和重量级锁 这四种锁是针对synchronized关键字提出的,在说这四种锁之前先来简单介绍一个重要的知识点:Mark Word Mark Word是保存在Java对象头中的数据,
线程执行到 monitorenter 指令时,将会尝试获取对象所对应的 monitor 的所有权,即尝试获得对象的锁。 2. Java对象头 锁存在Java对象头里。...Java对象头里的Mark Word里默认存储对象的HashCode,分代年龄和锁标记位。32位JVM的Mark Word的默认存储结构如下: ?...Java SE1.6为了减少获得锁和释放锁所带来的性能消耗,引入了“偏向锁”和“轻量级锁”,所以在Java SE1.6里锁一共有四种状态,无锁状态,偏向锁状态,轻量级锁状态和重量级锁状态,它会随着竞争情况逐渐升级...偏向锁的设置 关闭偏向锁:偏向锁在Java 6和Java 7里是默认启用的,但是它在应用程序启动几秒钟之后才激活,如有必要可以使用JVM参数来关闭延迟-XX:BiasedLockingStartupDelay.../how-the-java-virtual-machine-performs-thread-synchronization.html http://www.infoq.com/cn/articles/java-se
Java中的synchronized是非公平锁,ReentrantLock默认的lock方法采用的是非公平锁。...Java的ReentrantLock和synchronized都是可重入锁。 共享锁和独占锁 java并发包提供的加锁模式分为共享锁和独占锁。...ReadWriteLock 读写锁 为了提高性能,Java提供了读写锁,在读的地方使用读锁,在写的地方使用写锁,灵活控制。如果没有写锁的情况下,读是无阻塞的,在一定程度上提高了程序的执行效率。...Java读写锁有对应的接口java.concurrent.locks.ReadWriteLock,也有具体的实现ReentrantReadWriteLock。...补充相关知识: Java中的偏向锁,轻量级锁, 重量级锁解析 分段锁 ConcurrentHashMap,JDK7及之前实现同步的一种方式,对整个数组分段加锁,不是一种实际的锁。
上一篇简略说了一下Synchronized和Lock,今天就来说一下Java的锁机制。...Java的锁机制主要分为四种,分别是 (1)公平锁、非公平锁 (2)可重入锁 (3)自旋锁 (4)共享锁、独占锁 接下来一一说一下这四种锁 一、公平锁、非公平锁 (1)公平锁:指多个线程按照申请锁的顺序来获取锁...公平锁: ? 非公平锁: ? 二、可重入锁(又叫递归锁) (1)是指同一线程外层函数获取锁之后,内层递归函数仍然能持有锁继续运行。...(2)共享锁:指锁可以同时被多个线程持有。...对于ReentrantReadWriteLock来说,其由ReadLock(写锁)和WriteLock(读锁),其中写锁是独占锁,读锁是共享锁,保证高并发。读写,写读,写写的过程是互斥的。
竞争同一文件的两个线程可能在不同的 Java 虚拟机上,或者一个是 Java 线程,另一个是操作系统中的某个本地线程。...文件锁对其他的操作系统进程是可见的,因为 Java 的文件加锁直接映射到了本地操作系统的加锁工具(通过文件进行加锁)。...java文件锁要么独占,要么共享。 共享锁:允许多个线程对文件进行读操作。...) 是非阻塞式的,它设法获取锁,但如果不能获得,例如因为其他一些进程已经持有相同的锁,而且不共享时,抛出文件重叠锁异常【OverlappingFileLockException】。...OverlappingFileLockException 单个 Java 虚拟机在某个特定文件上所保持的锁定、不同 jvm 或者不同操作系统获取同一文件锁时,先拿到的获得锁,后获取的抛出文件重叠锁异常
image.png 乐观锁 VS 悲观锁 对于同一个数据的并发操作,悲观锁认为自己在使用数据的时候一定有别的线程来修改数据,因此在获取数据的时候会先加锁,确保数据不会被别的线程修改。...Java中,synchronized关键字和Lock的实现类都是悲观锁。...而乐观锁认为自己在使用数据时不会有别的线程修改数据,所以不会添加锁,只是在更新数据的时候去判断之前有没有别的线程更新了这个数据。如果这个数据没有被更新,当前线程将自己修改的数据成功写入。...乐观锁在Java中是通过使用无锁编程来实现,最常采用的是CAS算法,Java原子类中的递增操作就通过CAS自旋实现的。
最全Java锁详解:独享锁/共享锁+公平锁/非公平锁+乐观锁/悲观锁 在Java并发场景中,会涉及到各种各样的锁如公平锁,乐观锁,悲观锁等等,这篇文章介绍各种锁的分类: 公平锁/非公平锁 可重入锁...乐观锁适用于多读的应用类型,乐观锁在Java中是通过使用无锁编程来实现,最常采用的是CAS算法,Java原子类中的递增操作就通过CAS自旋实现的。...最全Java锁详解:独享锁/共享锁+公平锁/非公平锁+乐观锁/悲观锁 3.总之: 悲观锁适合写操作多的场景,先加锁可以保证写操作时数据正确。...最全Java锁详解:独享锁/共享锁+公平锁/非公平锁+乐观锁/悲观锁 3.典型应用: java jdk并发包中的ReentrantLock可以指定构造函数的boolean类型来创建公平锁和非公平锁(...Java线程锁 详细可以参考:高并发编程系列:4种常用Java线程锁的特点,性能比较、使用场景 本文标题:最全Java锁详解:独享锁/共享锁+公平锁/非公平锁+乐观锁/悲观锁 转载请保留页面地址:http
领取专属 10元无门槛券
手把手带您无忧上云