对于以下两个类,它们具有相同的Java字节码。
java版本:
java版本“ 1.8.0_181” Java™SE运行时环境(内部版本 1.8.0_181-b13)Java HotSpot(TM)64位服务器VM(内部版本25.181-b13,混合模式)
javac和javap版本:
1.8.0_181
我的疑问是
不应使用 synchronized 关键字的方法具有不同的字节码,正如我们在 synchronized block 中看到的那样,它们具有monitorenter
和monitorexit
,或让我们假设我不应该将同步块和同步
方法然后
JVM如何以不同方式处理这两种方法?
public class MySingleton1 {
private MySingleton1() {}
private static MySingleton1 ourInstance;
public static MySingleton1 getInstance() {
if (ourInstance == null) {
ourInstance = new MySingleton1();
}
return ourInstance;
}
}
和
public class MySingleton2 {
private MySingleton2() {}
private static MySingleton2 ourInstance;
public static synchronized MySingleton2 getInstance() {
if (ourInstance == null) {
ourInstance = new MySingleton2();
}
return ourInstance;
}
}
字节码如下:
$javac MySingleton1.java
$javap -c MySingleton1
$javac MySingleton2.java
$javap -c MySingleton2
各个文件的字节码:
Compiled from "MySingleton1.java"
public class MySingleton1 {
public static MySingleton1 getInstance();
descriptor: ()LMySingleton1;
Code:
0: getstatic #2 // Field ourInstance:LMySingleton1;
3: ifnonnull 16
6: new #3 // class MySingleton1
9: dup
10: invokespecial #4 // Method "<init>":()V
13: putstatic #2 // Field ourInstance:LMySingleton1;
16: getstatic #2 // Field ourInstance:LMySingleton1;
19: areturn
}
和
Compiled from "MySingleton2.java"
public class MySingleton2 {
public static synchronized MySingleton2 getInstance();
descriptor: ()LMySingleton2;
Code:
0: getstatic #2 // Field ourInstance:LMySingleton2;
3: ifnonnull 16
6: new #3 // class MySingleton2
9: dup
10: invokespecial #4 // Method "<init>":()V
13: putstatic #2 // Field ourInstance:LMySingleton2;
16: getstatic #2 // Field ourInstance:LMySingleton2;
19: areturn
}
我只是想增加我对java w.r.t.的了解。字节码。
如果我的方法错误或问题太琐碎,请让我作为评论。
除以下内容外,任何与文档相关的参考文献都非常受欢迎:
https://en.wikipedia.org/wiki/Java_bytecode
https://en.wikipedia.org/wiki/Java_bytecode_instruction_listings
答案 0 :(得分:5)
方法上的synchronized
修饰符被编译到方法标题中的ACC_SYNCHRONIZED标志中。它不影响生成的字节码指令; JVM看到该标志时会隐式添加进入和退出监视器的代码。
有关方法标头中标志的完整列表及其含义,请参见JVM specification。