我们知道,synchronized关键字是java中实现同步锁的关键字,那么,synchronized关键字是如何实现锁功能的呢?
其实,synchronized 关键字底层原理属于 JVM 层面。实现原理需要区分情况
第一种, synchronized 同步语句块的情况
![9840c83332f548ee8dcd8653d2c3ec9c.png](https://i-blog.csdnimg.cn/blog_migrate/fc8eed9298b4e3a8c3c85fdab69125a6.jpeg)
编辑搜图
请点击输入图片描述
我们可以通过 JDK 自带的 javap 命令查看 SynchronizedDemo 类的相关字节码信息:首先切换到类的对应目录执行
javac SynchronizedDemo.java
用于生成编译后的 .class 文件,然后执行
javap -c -s -v -l SynchronizedDemo.class
就可以看到如下所示
![ec39626687210d4a5910d77dea37378b.png](https://i-blog.csdnimg.cn/blog_migrate/31a862a2c006a8848e9344021bbf3a89.jpeg)
编辑搜图
请点击输入图片描述
从上面我们可以看出:
synchronized 同步语句块的实现使用的是 monitorenter 和 monitorexit 指令,其中 monitorenter 指令指向同步代码块的开始位置,monitorexit 指令则指明同步代码块的结束位置。 当执行 monitorenter 指令时,线程试图获取锁也就是获取 monitor(monitor对象存在于每个Java对象的对象头中,synchronized 锁便是通过这种方式获取锁的,也是为什么Java中任意对象可以作为锁的原因) 的持有权。当计数器为0则可以成功获取,获取后将锁计数器设为1也就是加1。相应的在执行 monitorexit 指令后,将锁计数器设为0,表明锁被释放。如果获取对象锁失败,那当前线程就要阻塞等待,直到锁被另外一个线程释放为止。
第二种 synchronized 修饰方法的的情况
![9a7ac7c2f041c50d1c76c55034ec1d47.png](https://i-blog.csdnimg.cn/blog_migrate/c32cbd46c517006d813f773f1e7fa5f4.jpeg)
编辑搜图
请点击输入图片描述
![299791b85e5fe50d416c969015857f20.png](https://i-blog.csdnimg.cn/blog_migrate/b2dbb0ca6150db5bcc127cdaa2af730e.jpeg)
编辑搜图
请点击输入图片描述
synchronized 修饰的方法并没有 monitorenter 指令和 monitorexit 指令,取得代之的确实是 ACC_SYNCHRONIZED 标识,该标识指明了该方法是一个同步方法,JVM 通过该 ACC_SYNCHRONIZED 访问标志来辨别一个方法是否声明为同步方法,从而执行相应的同步调用。