关于深入理解Java中的String:
https://www.cnblogs.com/xiaoxi/p/6036701.html
Java中String两种不同创建方式:
String str1 = "abc"; String str2 = new String("abc");
1、常量式创建
“我们知道字符串的分配和其他对象分配一样,是需要消耗高昂的时间和空间的,而且字符串我们使用的非常多。JVM为了提高性能和减少内存的开销,在实例化字符串的时候进行了一些优化:使用字符串常量池。每当我们创建字符串常量时,JVM会首先检查字符串常量池,如果该字符串已经存在常量池中,那么就直接返回常量池中的实例引用。如果字符串不存在常量池中,就会实例化该字符串并且将其放到常量池中。由于String字符串的不可变性我们可以十分肯定常量池中一定不存在两个相同的字符串。“
换句话说如果字符串常量池中存在“abc”,则直接指向它,否则就在字符串常量池中创建“abc”,并返回this。
2、对象式创建
这种创建形式跟我们一般创建对象的形式一样,创建完的对象存在堆中,指向存在栈中。所以str2的内存地址在栈中,指向堆中“abc”,而str1的内存地址指向字符串常量池。
但同时我们有个疑问。
String内部的char[]是final类型,意味着他不能被改变。由于String字符串的不可变性我们可以十分肯定常量池中一定不存在两个相同的字符串。那么上述两个str1和str2的字符串怎么内存地址不一样呢。可以这样理解。str1的内存地址指向字符串常量池是没错,str2的内存地址在栈中,指向堆中“abc”,然后堆中“abc”又引用了常量池中“abc”。但是str1和str2的内存地址确实不同。
那究竟哪种使用更好呢?
以下个人猜测。
根据如果在类中使用String str2 = new String("abc");这种对象式创建,随着出栈操作,str1的内存地址会被回收,而堆中对象空间也会被垃圾回收器回收,但相比str1,它多占了一份内存(即堆中str2)。适用于某个累内部多次引用,增加内存开销,来减少cpu开销。