1.先简单看一看JVM内存结构
方法区: 该区为各个线程共享,用于存储已经被虚拟机加载的类信息、常量、静态变量、即时编译器编译出来的代码等数据。
常量池就在这个区域
堆: Heap区被所有的线程共享,在虚拟机启动时创建。此区的功能就是存放对象实例,几乎所有的对象实例都是在这里分配内容。Heap区垃圾回收器管理的主要区域。
2.创建字符串对象
public class TestDemo {public static void main(String[] args) {String a = "1";String b = "1";String c = b;System.out.println(a == b);System.out.println(c == b);String d = new String("test");String e = new String("1");String f = new String(a);System.out.println(d == a);System.out.println(e == d);String g = "hello" + "tomorrow";String h = new String("hello") + new String("world");}
}
首先,看一下
String a = "1";String b = "1";String c = b;System.out.println(a == b);System.out.println(c == b);
控制台返回 true
分析上述代码的输出结果: 上述代码,只创建了一个对象
- 首先,jvm在编译阶段会判断方法区常量池中是否有 “1” 这个常量对象(
String a = "1";
)
如果有,a直接指向这个常量的引用
如果没有,就在常量池里创建这个常量对象 - 此过程并没有在堆中创建对象
String b = "1";
直接将 常量对象"1" 的地址交给了b,String c = b;
将 b 指向的 常量对象"1" 的地址交给了c- 当使用
==
判断时,都是在对比 常量池中的 常量对象"1" 的地址,故而相同,返回true
接着来看看
String d = new String("test");
分析: 上述代码,创建了两个对象
- 首先,jvm在编译阶段会判断方法区常量池中是否有 “test” 这个常量对象,没有就创建
- 其次,通过
new
在 堆 中创建 String对象,d 指向的就是这个String对象的地址
继续看
String e = new String("1");String f = new String(a);System.out.println(d == a);System.out.println(e == d);System.out.println(e == f);
控制台返回 false
分析上述代码的输出结果: 上述代码,创建了2个对象
- 首先,常量池中已经有了 “1” ,且 a 指向的也是 “1” 的地址
- 所以,此过程只在堆中用
new
创建两个 String 对象 - 虽然他们的字符串常量值都是 1,但是 e和f 指向的是两个不同的String对象的地址,所以返回值都为false
最后看:
String g = "hello" + "tomorrow";String h = new String("hello") + new String("world");
首先来分析String g = "hello" + "tomorrow";
只创建了1个对象
- jvm编译阶段过编译器优化后会把字符串常量直接合并成"hellotomorrow",所以最终只在常量池中创建了一个 “hellotomorrow” 常量对象
接着来看String h = new String("hello") + new String("world");
创建了6个对象
- 首先
new
创建了一个 StringBuilder() 对象 - 接着
h = new String("hello") + new String("world")
创建了 4 个对象(和上述创建过程相同) - 最终
new
创建了一个对象 String(“ab”)
3.补充
Integer m = 3;String s = m.toString(m);
注意: 此过程调用Object.toString()
,并没有在常量池中创建新的对象。