Synchronized
jvm是通过控制进出monitor对象来实现方法与代码块的同步,其中代码块同步是通过monitorenter和monitorexit来实现,monitorenter插入在代码块的开始位置,monitorexit插入在结束处或抛出异常处。方法同步通过在编译上加ACC_SYNCHRONIZED,然后使用monitorenter和monitorexit标记。
synchronized用的锁存在Java对象头里,对象头里包含有Mark Word,其中2bit用来存储锁的标记位,1bit用来储存是否为偏向锁,其他的储存对象的HashCode和分代年龄。
类锁与对象锁
对象锁:
-
synchronized(this){},锁住的是this指代的实例对象;
-
synchronized(类中的非静态成员变量){},锁住实体里的非静态变量;
-
在方法前直接加synchronized,锁住的是实例的非静态方法;
使用对象锁时,只有对new出来的同一个实例对象有效。
类锁:
- synchronized(person.class),直接将person这个类锁住;
- synchronized(静态成员变量),由于静态成员变量和类信息都只存放在方法区,所以锁住静态成员变量就相当于锁住了整个类;
- 在静态方法前直接加synchronized,锁住的也是整个类。
ReentrantLock
可重入锁,可中断锁,默认是非公平锁。底层基于AQS构建。
lock():首先判断AQS中的status标志位,为0表示未被占有,就把它置为1,并且设置当前线程为该锁的独占线程。
unlock():锁释放成功后,检查标志位若为SIGNAL就唤醒后面等待的线程
AQS:采用一个FIFO的双向队列,当一个节点被添加进队列时,需要用基于CAS的方法来保证线程安全性。每个节点都有一个标志位,-1为前节点为头节点的节点,-2为一般的等待节点。
AQS提供了getState(),setState(),compareAndSetState()来修改节点的状态
Synchronized和ReentrantLock的区别
-
Synchronized是Java的关键字,属于jvm层面的;
ReentrantLock是Java的一个类,是api层面的锁。 -
Synchronized不需要用户手动释放锁;
ReentrantLock需要用户手动释放锁,用lock(),unlock(),方法配合try/finally来使用。 -
Synchronized不可中断,只能等待当前线程运行完或者抛异常;
ReentrantLock可中断,可以通过设置超时时间或者调用interrpt()方法来实现。 -
Synchronized是非公平锁;
ReentrantLock可以在构造方法中传入变量,true为公平锁,false为非公平锁。 -
Synchronized只能随机唤醒一个或者唤醒所有线程;
ReentrantLock可以精确唤醒线程。