java多线程(三)Synchronized

、Synchronized 使用

多线程的情况下对共享变量的访问必须进行同步处理,否则会出现意想不到的结果。Synchronized关键字就是进行多线程同步用的。

Synchronized的作用主要有三个:(1)确保线程互斥的访问同步代码(2)保证共享变量的修改能够及时可见(3)有效解决重排序问题。从语法上讲,Synchronized总共有三种用法:

  (1)修饰普通方法,锁是当前实例对象。

public synchronized void method1(){}

  (2)修饰静态方法,锁是当前对象的Class对象。

public static synchronized void method1(){}

  (3)修饰代码块,锁是Synchonized括号里配置的对象。

synchronized (this) {}

锁的是对象,并不是代码,只要锁的对象是一样的,就只能有一个线程来访问该对象锁定的方法(不管有多少个代码用的是该对象锁,只有一个代码是可以被访问的);同理,只有锁的对象是不一样的,可以有多个线程一起来访问用一段代码。

二、Synchronized 原理

Synchonized代码块的反编译:

java多线程(三)Synchronized_第1张图片

关于这两条指令的作用,我们直接参考JVM规范中描述:

monitorenter

每个对象有一个监视器锁(monitor)。当monitor被占用时就会处于锁定状态,线程执行monitorenter指令时尝试获取monitor的所有权,过程如下:

1、如果monitor的进入数为0,则该线程进入monitor,然后将进入数设置为1,该线程即为monitor的所有者。

2、如果线程已经占有该monitor,只是重新进入,则进入monitor的进入数加1.

3.如果其他线程已经占用了monitor,则该线程进入阻塞状态,直到monitor的进入数为0,再重新尝试获取monitor的所有权。

monitorexit

指令执行时,monitor的进入数减1,如果减1后进入数为0,那线程退出monitor,不再是这个monitor的所有者。

其他被这个monitor阻塞的线程可以尝试去获取这个 monitor 的所有权。 

  通过这两段描述,我们应该能很清楚的看出Synchronized的实现原理,Synchronized的语义底层是通过一个monitor的对象来完成

其实wait/notify等方法也依赖于monitor对象,这就是为什么只有在同步的块或者方法中才能调用wait/notify等方法

否则会抛出java.lang.IllegalMonitorStateException的异常的原因。


Synchronized同步方法的反编译结果:

java多线程(三)Synchronized_第2张图片

  从反编译的结果来看,方法的同步并没有通过指令monitorenter和monitorexit来完成(理论上其实也可以通过这两条指令来实现),

不过相对于普通方法,其常量池中多了ACC_SYNCHRONIZED标示符。JVM就是根据该标示符来实现方法的同步的:当方法调用时,

调用指令将会检查方法的 ACC_SYNCHRONIZED 访问标志是否被设置,如果设置了,执行线程将先获取monitor,

获取成功之后才能执行方法体,方法执行完后再释放monitor。在方法执行期间,其他任何线程都无法再获得同一个monitor对象。 

实本质上没有区别,只是方法的同步是一种隐式的方式来实现,无需通过字节码来完成。








你可能感兴趣的:(java线程)