易错!!String类与常量池

一、String对象有两种创建方式:

//第一种
String str1 = "a";
//第二种
String str2 = new String("a");

两种方法是有差别的。

  • 第一种方式是在常量池中取对象:先检查字符串常量池中有没有"a",如果字符串常量池中没有,则创建一个,然后 str1 指向字符串常量池中的对象,如果有,则直接将 str1 指向"a";
  • 第二种方式是直接在堆内存空间创建一个新的对象。

再如:

String str3 = "a";
str1 与 str3 都指向常量池中的 “a”,而 str2 指向堆中新创建的 “a”。 在这里插入图片描述 记住一点:只要使用 new 方法,便需要创建新的对象。

二、String 类型的常量池的使用: - 直接使用双引号声明出来的 String 对象会直接存储在常量池中。 - 如果不是用双引号声明的 String 对象,可以使用 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 str = new String("abc"); 这句话创建了几个字符串对象? 答:1 或 2 个。如果常量池中已有 “abc”,只在堆空间创建一个 “abc”;如果常量池中没有 “abc”,将首先在池中创建,然后在堆空间中创建,因此将创建总共 2 个字符串对象。

String s1 = new String("abc");// 堆内存的地址值
String s2 = "abc";
System.out.println(s1 == s2);// 输出 false,因为一个是堆内存,一个是常量池的内存,故两者是不同的。
System.out.println(s1.equals(s2));// 输出 true,内容相同
四、字符串拼接

常量池中的对象可以直接拼接成一个新的字符串。两个字符串拼接的结果在堆中。

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
在这里插入图片描述

五、扩展:Integer 类 与 常量池

Integer 类默认创建数值[-128,127] 的相应类型的缓存数据。 1. Integer i1=40; Java 在编译的时候会直接将代码封装成 Integer i1=Integer.valueOf(40);,从而使用常量池中的对象。 2. Integer i1 = new Integer(40); 这种情况下会创建新的对象。

一个例子:

Integer i1 = 40;
Integer i2 = 40;
Integer i3 = 0;
Integer i4 = new Integer(40);
Integer i5 = new Integer(40);
Integer i6 = new Integer(0);

System.out.println("i1=i2 " + (i1 == i2));
System.out.println("i1=i2+i3 " + (i1 == i2 + i3));
System.out.println("i1=i4 " + (i1 == i4));
System.out.println("i4=i5 " + (i4 == i5));
System.out.println("i4=i5+i6 " + (i4 == i5 + i6));
System.out.println("40=i5+i6 " + (40 == i5 + i6));
结果:
i1=i2   true
i1=i2+i3 true
i1=i4 false
i4=i5 false
i4=i5+i6 true
40=i5+i6 true
解释:

语句 i4 == i5 + i6,因为 + 这个操作符不适用于 Integer 对象,首先 i5 和 i6 进行自动拆箱操作,进行数值相加,即 i4 == 40。然后 Integer 对象无法与数值进行直接比较,所以 i4 自动拆箱转为 int 值 40,最终这条语句转为 40 == 40 进行数值比较。

------ 本文结束感谢您的阅读 ------