synchronized的作用大概分为三种:
1.确保多线程互斥的访问多线程代码。2.保证变量的可见性。3.防止指令重排序。
那么synchronized 是如何实现这些功能的。
public class SynchronizedDemo { public void method() { synchronized (this) { System.out.println("Method 1 start"); } } }
以上面的Demo为例。对代码进行反编译。
结果如下:
关于monitorenter和monitorexit的作用,可以在jvm规范查找。
每个对象都有一个监视器(Monitor),当monitor被占用时就会被锁定,线程执行monitorenter指令就会尝试获取monitor的所有权。
获取过程如下:
1.如果monitor的进入数为0,则该线程进入该monitor,并把进入数设置为1,该线程即为monitor的所有者。
2.如果该线程已经占有该monitor,只是重新进入,则该monitor进入数 加一即可。
3.如果其他线程已经占有了该monitor,线程进入阻塞状态,直到该monitor进入数为0再重新获取。
反之monitorexit则是进入数减一了。
所以Synchronized的语义底层是通过一个monitor的对象来完成,其实wait/notify等方法也依赖于monitor对象,这就是为什么只有在同步的块或者方法中才能调用wait/notify等方法,否则会抛出java.lang.IllegalMonitorStateException的异常的原因。