1、原始构成
1.1、是关键字
synchronized属于JVM层面
- monitorenter(进来):底层是通过monitor对象来完成的,wait/notify等方法也依赖于monitor对象,只有在同步块或方法中才能调用wait/notify等方法
- monitorexit(出去):这个在底层会出现两次,分别是正常退出和异常退出。相当于双重保险,主要是为了保证不产生死锁。
1.2、lock是具体类
java.util.concurrent.locks.Lock,是api层面的锁。
2、使用方法
2.1、synchronized
不需要用户去手动释放锁,当synchronized代码执行完成后系统会自动让线程释放对锁的占用
2.2、ReentrantLock
需要用户去手动释放锁,若没有主动释放锁,就有可能出现死锁现象。需要lock()和unlock()方法配合try/finally语句块来完成
3、等待是否可中断
3.1、synchronized
不可中断,除非抛出异常或者正常运行完成
3.2、ReentrantLock
可中断,
- 设置超时方法tryLock(Long timeout , Timeout unit )
- lockInterruptibly()放在代码块中,调用interrupt()方法可中断
4、加锁是否公平
4.1、synchronized
非公平锁
4.2、ReentrantLock
两者都可以,默认非公平锁,构造方法可以传入boolean值,true为公平锁,false为非公平锁
5、锁绑定多个条件Condition
4.1、synchronized
没有
4.2、ReentrantLock
用来实现分组唤醒需要唤醒的线程们,可以精确唤醒,而不是像synchronized要么随机唤醒一个线程,要么话唤醒全部线程