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() ,并没有在常量池中创建新的对象。
|