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的异常的原因。

posted on 2017-07-05 16:42  知己一生  阅读(502)  评论(0编辑  收藏  举报