文章目录
前言
- 一定要看注释!!!
一、String几种创建方式
a. 创建方式1:String s1 = “a”;
直接将a放入串池中,创建0个或1个对象(0个是因为可能之前已存在串池了)
b. 创建方式2: String s2 = new String(“b”);
在堆中先创建对象,再查看串池中是否存在b,存在就不创建,不存在就创建,都会返回串池的引用,创建1个或2个对象
c. 创建方式3:String s3 = s1 + s2;
在运行期间才确认,只在堆中创建ab对象,并不会放入串池,创建0个或1个对象
d. 创建方式4:String s4 = “a” + “b”;
编译优化,等价于String s4 = “ab”,创建0个或1个对象
e. 创建方式5:String s5 = new String(“a”) + new String(“b”);
和s3的创建方式类似,也是在运行期间才确认,只在堆中创建ab对象,并不会放入串池,创建了1个对象(不包括串池中的a、b和堆中的两个匿名对象)
二、堆、字符串常量池什么关系?
在jdk1.7以后字符串常量池是在堆中,如下图:
三、String类的intern()方法
- 在jdk1.7及以后,如果该String对象不存在串池中,调用intern方法是将字符串直接放入串池;简单来说就是本来在堆中的对象到字符串常量池。举个例子:
String s1 = new String("a") + new String("b");//s1只存在堆中
String s2 =s1.intern();//将ab放入串池,引用s1也指向了串池中的ab
System.out.println( s1 == s2 ); //结果是true
- 在jdk1.7以前,intern方法是拷贝堆中的对象放入串池,有别于移动。同上面例子:
String s1 = new String("a") + new String("b");//s1只存在堆中
String s2 = s1.intern();//调用intern方法发现串池没有ab就把堆中的拷贝到串池中,返回引用时串池的,同时不该变s1的指向
System.out.println( s1 == s2 ); //结果是flase
- 总结:其实把两个行为理解为剪切-粘贴和复制-粘贴比较好,剪切-粘贴地址肯定改变,引用就改变了,而复制-粘贴不会!