synchronized 的实现方式
synchronized
的类型可以分为两种:
- synchronized method
- synchronized block
两者的实现方式是不一样的,jvm 规范中写道,编译后的 synchronized method 会有一个 ACC_SYNCHRONIZED 的 flag,也就是说当 jvm 的方法调用指令(the method invocation instruction)从 the run-time constant pool 中查找到这个 method 的时候,已经知道它是一个synchronized method,所以锁操作是由方法调用以及返回指令来控制的。
而 synchronized block 的锁是由 monitorenter
和 monitorexit
这两个指令来控制。
可以通过 javap 命令来“反汇编”一下 class 文件。
类和对象
首先要了解的就是类究竟是怎么来的。
JVM 拿到编译器编译好的 class 文件后,首先会把文件载入到内存中,class 文件当然会有自己的格式,所以需要由 ClassLoader 来解析文件的内容,这个解析出来的内容会用一个 Class
类的实例 - Class object 来表示,这个 object 可以通过 Java 的 ClassName.class
来获取。
也就是说,Class object 是一个 Class
类型的实例(instance),而对象是一个 ClassName 的 instance。Class 和 ClassName都是类型,ClassName是由 class
关键字定义的,而Class是内置类型。
因此成员方法的synchronized method 就等价于 synchronized (this) block,即下面两种方式是等价的。
public synchronized void fun1() {
// do something here
}
public synchronized void fun2() {
synchronized (this) {
// do something here
}
}
成员方法是属于 this
,而静态方法是属于 Class Object,那么静态方法的 synchronized method 也就等价于下面这种形式的 synchronized block 了。
public static synchronized void fun2() {
synchronized (ClassName.class) {
// do something here
}
}