String的两种创建方式的区别
String str1 = "abc";
String str2 = new String("abc");
sout(str1 == str2);//false
其中,第一种方式是从常量池中获取对象,第二种方式是直接在堆内存中创建一个新的对象。
String 类型的常量池
它的主要使用方法有两种:
- 直接使用双引号声明出来的 String 对象会直接存储在常量池中。
- 如果不是用双引号声明的 String 对象,可以使用 String 提供的 intern 方法。String.intern() 是一个 native 方法,它的作用是:如果运行时常量池中已经包含一个等于此 String 对象内容的字符串,则返回常量池中该字符串的引用;如果没有,则在常量池中创建与此 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个。
吃水不忘挖井人: |