1.看看下面这段代码执行结果是什么?
String a = "ab";
String b = "a" + "b";
System.out.println((a == b)); //它的执行结果为true;
2.下面这段代码的执行结果是什么?
String a = "ab";
String bb = "b";
String b = "a" + bb;
System.out.println((a == b)); //执行结果为false
以上两个执行结果的原因分析:
String b = "a" + "b";编译器将这个"a" + "b"作为常量表达式,在编译时进行优化,直接取结果"ab",这样这个问题退化
String a = "ab";
String b = "ab";
System.out.println((a == b));
然后根据3的解释,得到结果true
javap工具执行查看字节码如下:
0: ldc #16; //String ab
2: astore_1
3: ldc #16; //String ab
5: astore_2
6: getstatic #18; //Field java/lang/System.out:Ljava/io/PrintStream;
9: aload_1
10: aload_2
11: if_acmpne 18
14: iconst_1
15: goto 19
18: iconst_0
19: invokevirtual #24; //Method java/io/PrintStream.println:(Z)V
22: return
通过 0: ldc #16; //String ab和 3: ldc #16; //String ab ,知道该java文件通过编译器编译后都变为 String b = "ab"; 和作者分析的一样。
===========================
对于第二个程序这个好理解,"a" + bb中的bb是变量,不能进行优化。javap查看字节码如下:
0: ldc #16; //String ab
2: astore_1
3: ldc #18; //String b
5: astore_2
6: new #20; //class java/lang/StringBuilder
9: dup
10: ldc #22; //String a
12: invokespecial #24; //Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V
15: aload_2
16: invokevirtual #27; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
19: invokevirtual #31; //Method java/lang/StringBuilder.toString:()Ljava/lang/String;
22: astore_3
23: getstatic #35; //Field java/lang/System.out:Ljava/io/PrintStream;
26: aload_1
27: aload_3
28: if_acmpne 35
31: iconst_1
32: goto 36
35: iconst_0
36: invokevirtual #41; //Method java/io/PrintStream.println:(Z)V
39: return
String+String的操作是在运行时进行的,则会产生新的对象,而不是直接从jvm的string池中获取。
===============================
再看下面两个例子:
String a = "ab";
final String bb = "b";
String b = "a" + bb;
System.out.println((a == b)); //result = true
String a = "ab";
final String bb = getBB();
String b = "a" + bb;
System.out.println((a == b)); //result = false
private static String getBB() {
return "b";
}
查看下面的字节码如下:
0: ldc #16; //String ab
2: astore_1
3: invokestatic #18; //Method getBB:()Ljava/lang/String;
6: astore_2
7: new #22; //class java/lang/StringBuilder
10: dup
11: ldc #24; //String a
13: invokespecial #26; //Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V
16: aload_2
17: invokevirtual #29; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
20: invokevirtual #33; //Method java/lang/StringBuilder.toString:()Ljava/lang/String;
23: astore_3
24: getstatic #36; //Field java/lang/System.out:Ljava/io/PrintStream;
27: aload_1
28: aload_3
29: if_acmpne 36
32: iconst_1
33: goto 37
36: iconst_0
37: invokevirtual #42; //Method java/io/PrintStream.println:(Z)V
40: return
分析:0: ldc #16; //String ab 变量a的值直接指向常量池中的‘ab’,
通过下面的分析可知 变量2通过StringBuilder进行string操作。
分享到:
相关推荐
第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: ...
深入JVM内核—原理、诊断与优化视频教程 深入JVM内核—原理、诊断与优化视频教程
深入JVM内核—原理、诊断与优化视频教程 深入JVM内核—原理、诊断与优化视频教程
深入jvm 内核-原理,诊断于优化视频教程.算是比较基础的
深入JVM内核—原理、诊断与优化视频教程———目前,Java是更为流行的编程语言之一,它的基础平台就是JVM。除了Java,如JRuby、Scala、Clojure等语言也运.
1、java虚拟机的基本介绍。 2、字节码的执行 3、常用的jvm参数配置 4、算法和种类 5、gc参数配置 6、类加载器 7、性能监控工具 8、jvm堆栈分析
当前,以Hadoop、Spark为...这些大数据处理框架采用分布式架构,使用Java、Scala等面向对象语言编写,在集群节点上以Java虚拟机(JVM)为运行时环境执行计算任务,因此依赖JVM的自动内存管理机制来分配和回收数据对象.
jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识...
Java助力需要jvm学习及优化与性能瓶颈分析参考
1 什么是性能优化\ 2 性能测试与优化\ 3 性能优化JVM篇\ 4 性能优化Tomcat篇\ 5 性能优化mysql篇\
JVM 深入学习教程深入分析JVM教程!jvm 内存原型,优化等等
jvm参数优化后,tomcat稳定可靠,附件为通过长时间在线测试的配置参数文件
熟悉和掌握JVM平台有着重要的实用价值和意义。 在本课程中个,将详细介绍JVM的基本原理、组成以及工作方式,并配合实际案例,介绍相关的调优技巧。 课程大纲: 第一课 初识JVM JVM分类 Java语言规范 JVM规范 ...
JVM内存模型深度剖析与优化
invokespecial:用于调用实例构造器方法、私有方法和父类中的方法。 invokevirtual:用于调用所有的虚方法。 invokeinterface:用于调用接口方法,会在运行时再确定一个实现该接口的对象。 invokedynamic:...
深入JVM内核—原理、诊断与优化视频教程 深入JVM内核—原理、诊断与优化视频教程
实战JAVA虚拟机 JVM故障诊断与性能优化.pdf (无书签.低分放送) 实战JAVA虚拟机 JVM故障诊断与性能优化.pdf (无书签.低分放送) 实战JAVA虚拟机 JVM故障诊断与性能优化.pdf (无书签.低分放送)