JUC-锁
乐观锁和悲观锁
悲观锁
当一个线程在操作资源的时候,会悲观的认为有其他的线程会来抢占该资源,因此会在操作资源前进行加锁,避免其他线程抢占。
Synchronized关键字和Lock实现类就是悲观锁。
显示的锁定资源后再对资源进行操作。
使用场景:
- 适合写操作多的场景。先加锁能够保证写操作时数据正确
本质:
加锁去操作同步资源。
乐观锁
当一个线程去操作资源的时候,会乐观的任务其他线程不会来抢占资源,因此不会加锁。
java中通过无锁编程来实现,只是在对数据进行修改的时候,判断其他线程是否对该数据进行修改过
- 如果没有修改过,该线程直接修改数据。
- 如果修改过,该线程则根据不同的实现方式执行不同的操作,比如放弃修改,重试抢锁等等。
原子操作类那些底层的是CAS(Compare And swap)算法,也就是乐观锁。
判断规则:
- 版本号机制Version(每修改一次版本号递增,当前版本号是最大的,可以直接修改。不是最大的,意味着别人修改过了,我的修改要重新处理)
- 最常采用的是CAS算法(后面会详细讲,这里略)
使用场景:
乐观锁适合读操作多的场景,不加锁读操作性能大幅提升
本质:
无锁去操作同步资源。
乐观锁和悲观锁举例
悲观锁:Synchronized和Lock的实现类
乐观锁:原子操作的类AtomicInteger, LongAdder
Synchronized
阿里加锁规范
高并发时,同步调用时需要考虑加锁性能损耗。能用无锁数据结构就用无锁数据结构。能用块锁,就不要锁方法体。能用对象锁,就不要用类锁。
(尽可能让锁的代码块尽可能小,避免锁造成不必要的性能开销)
Synchronized三种作用方式
作用于实例方法:当前实例加锁,进入实例前要获取当前实例的锁对象。
作用于代码块:对括号里的对象进行加锁。
作用于静态方法(类方法):对当前类加锁,进去同步代码前要获得当前对象的锁。
Synchronized作用于非静态方法和静态方法的区别(重要)
类中Synchronized修饰非静态方法(对象锁)
- 加的锁为this对象锁。
- 一个对象只有一把对象锁,因此多个线程执行一个对象的非静态同步方法时,存在竞争关系。先获得对象锁的线程先执行。(不同对象不会有竞争)
- 不同对象有不同的对象锁,线程如果持有不同对象锁,线程间无竞争的关系。
类中Synchronized修饰静态方法(类锁)
- 加的锁为类锁。
- 先获得类锁的线程先执行。多个线程执行同一个类模板的不同对象的静态同步方法的时候,存在竞争关系。先获得类锁的线程先执行。(同一个对象会竞争,不同对象也会竞争)
- 不同类有不同的类锁,线程如果持有不同的类锁,线程间无竞争关系
- 一个对象的类锁和对象锁是不同的锁。一个线程持有类锁,一个线程持有对象锁,线程间无竞争关系。
类中无Syncronize修饰的方法(和锁无关)
线程执行该方法不需要获得锁,直接执行就行了。
代码
1 | class Phone //资源类 |
字节码角度分析Synchronized
查看反汇编:
1 | javap -c *.java// -c对代码进行反汇编。 -v (verbose)输出附加信息行号,本地变量表,反汇编 |
synchronized同步代码块
- 实现使用的是monitorenter和monitorexit。monitorenter代表获得锁对象,monitorexit代表释放锁对象。
- 通常情况下,一个monitorenter对应两个monitorexit,正常情况下,从第一个monitorexit释放锁。异常情况下,从第二个monitorexit释放锁。
synchronized普通同步方法
调用指令时,先检查ACC_SYNCHRONIZED(Access)标志是否被设置了,如果该方法有这个标志,代表是同步方法,访问的时候要获取锁对象。
方法完成时(无论是否正常介数)释放锁。
synchronized静态同步方法
调用指令时,ACC_STATIC,和ACC_SYNCHRONIZED标志。第一个表示是否静态方法,第二个表示是否同步方法。
反编译Synchronized锁是什么
为什么任何一个对象都可以成为锁?
Java虚拟机支持方法级
什么是管程?
管程(Monitor):可以看做是一个功能模块,他将共享变量和对共享变量的操作封装起来。进程可以调用管程实现进程间的并发控制。
同步指令实现?
Java虚拟机支持方法级的同步和方法内部指令序列的同步,这两种同步结构都是由管程(Monitor或者称为锁)来实现的。
- 方法级的同步:通过读取ACC_SYNCHRONIZED判断是否是同步方法,如果是同步方法,执行线程要求必须持有管程(锁)。执行完毕后释放锁。
- 方法内部指令序列的同步:同步一段指令序列是通过synchronized方法块来表示。java虚拟机指令集中的monitorenter和monitorexit指令实现的。
Monitor的实现 OjectMonitor
每个对象都关联一个ObjectMonitor锁对象。他有一些属性来保证该资源的同步安全。
ower: 持有该锁的线程
waitset:存放处于wait状态的线程队列
entrylist:存放等待锁的线程队列
recursions(递归):锁的重入次数
count: 记录该线程获取锁的次数。
公平锁和非公平锁
公平锁(先来先得)
多个线程按照线程请求锁的先后顺序获取锁。默认都是非公平锁,公平锁需要设置。
1 | Lock lock = new ReentrantLock(true);/l/true表示公平锁,先来先得 |
执行流程:
获取锁的时候,会将线程自己添加到等待队列中并休眠。当线程使用完锁之后,会去唤醒等待队列首部的线程。线程的休眠和恢复需要从用户态转换为内核态,线程切换是比较慢的,所以公平锁的执行较慢。
非公平锁(随机获得锁,默认)
每个线程获取到锁的顺序是随机的,并不会按照先来先得的顺序。所有的线程会竞争获取锁。
执行流程:
当线程申请锁时,会通过CAS尝试获取锁。如果获取成功,就持有锁对象。如果获取失败,就进入等待队列。好处是不用遵循先到先得的原则,避免了线程的休眠和恢复过程,执行更快。
使用场景
默认是非公平锁。能够让程序执行更快(追求效率)。
非公平锁可能造成线程饿死的情况。
可重入锁(递归锁)
定义
可重入锁又叫递归锁。一个线程在外部方法中获取到锁的时候。在进入内部方法需要获取锁的时候,线程会自动获取到该锁。而不会阻塞。
种类
隐式锁(Synchronized关键字修饰的):
线程在外部获取锁之后,内部自动获取到锁。
实现原理
每个锁对象ObjectMonitor都有一个count计数器和ower持有该锁对象的线程。
当执行monitorenter的时候:会看count计数器是否为0,如果为0说明该锁对象没有被其他线程占有,将count计数器+1,将ower设置为当前的线程。如果不为0,该线程需要等待。
当执行monitorexit的时候:会将count计数器减一,count为0代表可以释放。将ower清空。
显式锁(Lock实现类)
1 | lock.lock();//加锁 |
加锁和释放锁的次数要一样,不然会导致该线程一直持有锁。其他线程无法获取锁。
死锁
一个线程持有某个锁对象,有需要申请其他的锁对象。其他锁对象被另一个线程占有。在无外力干扰的情况下,一直处于僵持状态。
举例: A线程持有obj1锁对象,申请obj2锁对象。B线程持有obj2锁对象,申请obj1锁对象。A,B线程均被阻塞住,处于僵持状态。
手写一个死锁的例子
1 | final Object obj1 = new Object(); |
运行结果:
检测死锁
第一种方式命令行jps+jstack
jps查看死锁线程编号
1 | jps -l |
jstack 查看当前时刻的线程快照
1 | jstack 13992 |