对方法块同步:
每个对象都有一个监视器锁(monitor)。
线程执行monitorenter指令时尝试获取monitor的所有权。过程如下:
1.如果monitor的进入数是0,则该线程进入monitor并将进入数设置为1。该线程即为monitor的所有者。
2.如果线程已经占有该monitor,只是重新进入,则monitor进入数加1。
3.如果其他线程已经占用了monitor,则该线程进入阻塞状态直至monitor的进入数为0,然后尝试获取monitor的所有权。
线程执行monitorexit指令退出monitor。过程如下:
1.执行monitorexit指令的线程必须是monitor的拥有者。
2.线程执行该指令,monitor的进入数递减,当monitor进入数为0时该线程退出monitor并不再是monitor的所有者,其他的被这个monitor阻塞的线程可以尝试去获取这个monitor的所有权。
注:wait/notify等方法也依赖于monitor对象,所以只有在同步的块或者方法中才能调用wait/notify等方法,否则会抛出java.lang.IllegalMonitorStateException的异常
对方法同步:
方法的同步并没有通过指令monitorenter和monitorexit来完成(理论上其实也可以通过这两条指令来实现),不过相对于普通方法,其常量池中多了ACC_SYNCHRONIZED标示符。
JVM就是根据该标示符来实现方法的同步的:当方法调用时,调用指令将会检查方法的 ACC_SYNCHRONIZED 访问标志是否被设置,如果设置了,执行线程将先获取monitor,获取成功之后才能执行方法体,方法执行完后再释放monitor。在方法执行期间,其他任何线程都无法再获得同一个monitor对象。 其实本质上没有区别,只是方法的同步是一种隐式的方式来实现,无需通过字节码来完成。