String的两种创建方式
1. 常量式创建:
String str1 = "abcd";
当我们创建字符串常量时,JVM会首先检查字符串常量池,如果该字符串已经存在常量池中,那么就直接返回常量池中的实例引用。如果字符串不存在常量池中,就会实例化该字符串并且将其放到常量池中。由于String字符串的不可变性我们可以十分肯定常量池中一定不存在两个相同的字符串。
上面这行代码可以理解为:先检查字符串常量池中有没有"abcd"
,如果字符串常量池中没有,则创建一个,然后 str1
指向字符串常量池中的对象。如果有,则直接将 str1
指向"abcd"
;
2. 对象式创建:
String str2 = new String("abcd");//堆中创建一个新的对象
String str3 = new String("abcd");//堆中创建一个新的对象
这种创建形式跟创建对象的形式一样,创建完的对象存在堆中,指向存在栈中。所以str2
和str3
的内存地址在栈中,指向堆中“abcd”
,而str1
的内存地址指向字符串常量池。
这两种不同的创建方法是有差别的:
- 第一种方式是在常量池中拿对象;
- 第二种方式是直接在堆内存空间创建一个新的对象。
System.out.println(str1==str2);//false
System.out.println(str2==str3);//false
记住一点:只要使用 new
,便需要创建新的对象。
再给大家一个图应该更容易理(图片来源:https://www.journaldev.com/797/what-is-java-string-pool)
String 类型的常量池
String 类型的常量池比较特殊。它的主要使用方法有两种:
- 直接使用双引号声明出来的 String 对象会直接存储在常量池中。
- 如果不是用双引号声明的 String 对象,可以使用 String 提供的
intern
方法。
补充:String.intern()
String.intern()
是一个 Native 方法。它的作用是:
- 如果运行时常量池中已经包含一个等于此 String 对象内容的字符串,则返回常量池中该字符串的引用;
- 如果没有,JDK1.7之前(不包含1.7)的处理方式是在常量池中创建与此 String 内容相同的字符串,并返回常量池中创建的字符串的引用。JDK1.7以及之后的处理方式是在常量池中记录此字符串的引用,并返回该引用。
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 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
尽量避免多个字符串拼接,因为这样会重新创建对象。如果需要改变字符串的话,可以使用 StringBuilder 或者 StringBuffer。
ps:如果想了解8种基本类型的包装类和常量池,可以参考我的另外一篇博客:
https://blog.csdn.net/weixin_43901865/article/details/112566955