面试必问之synchronized与Lock的区别及底层实现

1、synchronized 是Java内置的关键字,使用后会自动释放锁,

Lock是java.util.concurrent.Locks 包下的一个接口,必须要手动释放。特别是在发生异常时,需要在 finally 块中进行手动释放,否则会发生死锁行为

synchronized 代码清单

Lock 代码清单
面试必问之synchronized与Lock的区别

2、Lock可响应中断,而synchronized 不能响应中断,并且Lock提供了更丰富的方法实现;例如

Lock() ; //获取锁

tryLock(); //获取锁

tryLock(long time, TimeUnit unit); //在一定时间单位内等待

lockInterruptibly(); //获取锁,可响应中断(AB线程同时获取锁,A得到后,B进行等待,则B会被Tread.interrupt()方法中端并可去执行其他的代码逻辑,而synchronized无法被中端)

unlock(); //释放锁

***以下对lockInterruptibly 进行简单的验证,代码结构糙,只为验证***

面试必问之synchronized与Lock的区别

3、synchronized 是非公平锁,即不能保证等待锁线程的顺序,

Lock的实现 ReentrantLock 可通过实例化true or false 的构造参数实现公平锁和非公平锁,默认为非公平锁

面试必问之synchronized与Lock的区别

ReentrantLock的构造参数

4、ReentrantLock是唯一实现了Lock接口的类,并且ReentrantLock提供了更多的方法

lck.isFair();
lck.isLocked();
lck.getHoldCount();
lck.getQueueLength();
lck.wait();
...

5、synchronized无法判断是否获取锁的状态,Lock可以判断是否获取到锁;

6、Lock锁适合大量同步的代码的同步问题,synchronized锁适合代码少量的同步问题。

7、都是可重入锁:在执行对象中所有同步方法不用再次获得锁

8、synchronized是一个悲观锁,Lock是一个乐观锁(底层基于volatile和cas实现),

二、底层实现

1、synchronznized映射成字节码指令就是增加两个指令:monitorenter、monitorexit,

当一条线程执行时遇到monitorenter指令时,它会尝试去获得锁,如果获得锁,那么所计数器+1(为什么要加1,因为它是可重入锁,可根据这个琐计数器判断锁状态),如果没有获得锁,那么阻塞,

当它遇到一个monitoerexit时,琐计数器会-1,当计数器为0时,就释放锁

(tips:节码中出现的两个monitoerexit指令的原因是:一个正常执行-1,令一个异常时执行,这两个用goto的方式只执行一个)
2、Lock底层则基于volatile和cas实现

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值