String的两种创建方式的区别
String str1 = "abc"; String str2 = new String("abc"); sout(str1 == str2);//false
其中,第一种方式是从常量池中获取对象,第二种方式是直接在堆内存中创建一个新的对象。
String 类型的常量池
它的主要使用方法有两种:
String s1 = new String("计算机"); String s2 = s1.intern(); String s3 = "计算机"; System.out.println(s2);//计算机 System.out.println(s1 == s2);//false,因为一个是堆内存中的String对象一个是常量池中的String对象, System.out.println(s3 == s2);//true,因为两个都是常量池中的String对象
String 字符串拼接
String str1 = "str"; String str2 = "ing"; String str3 = "str" + "ing";//常量池中的对象 String str4 = str1 + str2; //在堆上创建的新的对象 String str5 = "string";//常量池中的对象 System.out.println(str3 == str4);//false System.out.println(str3 == str5);//true System.out.println(str4 == str5);//false
常见问题
问题一:
String s1 = new String("abc"); // 这句话创建了几个对象?
解答:
String s1 = new String("abc");// 堆内存的地值值 String s2 = "abc"; System.out.println(s1 == s2);// 输出false,因为一个是堆内存,一个是常量池的内存,故两者是不同的。 System.out.println(s1.equals(s2));// 输出true
解释:
先有字符串 “abc” 放入常量池,然后 new 了一份字符串 “abc” 放入 Java 堆(字符串常量 “abc” 在编译期就已经确定放入常量池,而 Java 堆上的 “abc” 是在运行期初始化阶段才确定),然后 Java 栈的 str1 指向 Java 堆上的 “abc”。
问题二:
String s = "a" + "b" + "c" + "d" + "e";
问此语句共创建了几个对象?
答案:就创建一个。
String s = "a" + "b" + "c" + "d" + "e";
赋值符号右边的"a"、"b"、"c"、"d"、"e"都是常量,对于常量,编译时就直接存储它们的字面值而不是它们的引用,在编译时就直接讲它们连接的结果提取出来变成了"abcde"。该语句在class文件中就相当于String s = "abcde" ,然后当JVM执行到这一句的时候, 就在String pool里找,如果没有这个字符串,就会产生一个。
问题三:
但是如果改成 String s = a+b+c+d+e,又是几个呢?
我的答案是3个对象,但只有一个String对象。
由于编译器的优化,最终代码为通过StringBuilder完成:
StringBuilder builder = new StringBuilder(); builder.append(a); builder.append(b); builder.append(c); builder.append(d); builder.append(e); String s = builder.toString();
生成的三个对象为:
StringBuilder
new char[capacity]
new String(value,0,count);
如果说String对象,则为1个。
吃水不忘挖井人:
|