加锁和释放锁的原理

Posted steakliu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了加锁和释放锁的原理相关的知识,希望对你有一定的参考价值。

加锁和释放锁的原理

  当方法执行完后或者抛出异常后,都会释放锁

技术图片

method1等价于method2,因为method1执行完以后会自动释放锁,而么method2使用lock.unlock()释放锁

 

反编译看monitor指令

被 synchronized 关键字修饰的方法、代码块,就是 monitor 机制的临界区

进入锁和释放锁是基于monitor来实现的 同步方法和同步代码块,monitor有两个指令,monitorenter会插入到同步代码块的位置,monitorexit会插入到方法块结束和退出的时候,可能有多个monitorexit对应一个monitorenter,因为退出可以是方法结束或者抛出异常

技术图片

 

通过反编译(把.class文件转为.java)

编译:

技术图片

反编译:

技术图片

技术图片

Monditorenter和Monditorexit

实际上一个对象和一个monitor相关联,一个monitorde lock锁只能被一个线程再同一时间获得

Monditorenter的三种情况

 1.当monitor计数器为0,代表没有被获得,然后线程立马获得该Monitor,并把计数器加1,当别的线程想进来,但是看到计数器为1,就代表已经被其他线程占有,就只有等待

 

2.如果Monitor已经拿到了锁的所有权,又重入了锁,则monitor会累加

 

3.如果monitor被其他线程占用了,当我想去获取的时候,不能获取到,只能处于阻塞状态,当计数器为0才能再次尝试去获取锁

Monditorexit的作用是释放锁,前提是要先拥有此锁,原理是将计数器减1,减1如果monitor为0,则对象不再拥有对锁的所有权,就是解锁,如果不是1,则证明是重入进来的,继续持有锁,

 

可重入原理

 

技术图片

 

 可见性原理

 

 技术图片

 

以上是关于加锁和释放锁的原理的主要内容,如果未能解决你的问题,请参考以下文章

悲观锁和乐观锁

java基础--25.多线程的改进--Lock显式地加锁和释放锁

高效编程之互斥锁和自旋锁的一些知识

并发容器之写时拷贝的 List 和 Set

Redis分布式锁的正确加锁与解锁方式

AQS共享锁和独占锁