String新建,拼接的内存问题
去年老师讲Java基础的时候,讲解Java中String的时候,让自己去研究测试一下新建String类的一些不同.当时在没意,觉得不就是一个引用变量嘛,现在因为要实习复习一下基础内容,发现它能很好的解释Java的栈内存,堆内存和堆内存中的常量池.
我们知道String属于Java的引用变量类型,新建它的时候可以使用一下两种方式:
String str1 = "hello java";
String str3 = new String("hello java");
其实对于字符串的新建,String str1和String str3过程其实是在栈内容中开辟了一个变量,存放堆内容的地址,如果仅仅是先声明后复制,str3中存放的就是null,无指向.
我们一般开辟变量的同时都新建了对象,如:String str3 = new String(“hello java”),此时Java会在堆内存中开辟一片空间,使用str3指向它.
如果我们重复的使用new 新建字符串new String(“hello java”),即使字符串内容是一样,Java也会在堆内存中重复的新建对象.可以参照下面代码str3和str4及输出结果;
并且使用"+"拼接字符串,即使使用同一个变量来输出地址,拼接后的字符串也是在堆内存中新建的对象,可以参照下面代码str1 和拼接后的输出结果;
接下来我们看字符串的另一种创建方式String str1 = “hello java”,使用此种创建方式,Java会在堆内存中的常量池中,寻找是否有此数据,如果没有会建立一个,在下一次新建时,直接拿来使用.可以参照下面str1和str2及输出结果;
测试代码:
public class demo1 {
public static void main(String[] args) {
String str1 = "hello java";
System.out.println(String.class.getName() + "@" + Integer.toHexString(System.identityHashCode(str1)));
str1 = str1 + "js";
System.out.println(String.class.getName() + "@" + Integer.toHexString(System.identityHashCode(str1)));
String str2 = "hello java";
System.out.println(String.class.getName() + "@" + Integer.toHexString(System.identityHashCode(str2)));
String str3 = new String("hello java");
System.out.println(String.class.getName() + "@" + Integer.toHexString(System.identityHashCode(str3)));
String str4 = new String("hello java");
System.out.println(String.class.getName() + "@" + Integer.toHexString(System.identityHashCode(str4)));
}
}
运行结果:
java.lang.String@5cad8086
java.lang.String@6e0be858
java.lang.String@5cad8086
java.lang.String@61bbe9ba
java.lang.String@610455d6