Java使用 “ ” 称为字符串常量,为了提高程序的运行速度,节省空间,JVM会维护一个字符串常量池。当字符串常量第一次出现,则产生新对象并将该对象置入常量池中,后续如果再出现该字符串常量,不会产生新对象,直接复用常量池中的已有对象。
//第一次出现,产生新对象,入池
String s1="hello";
//第二次出现,不会产生新对象
String s2="hello";
//直接赋值法
String s1="hello";
String s2="hello";
//构造方法产生新对象
String s3=new String("hello");
String s4=new String("hello");
分析:对于s1和s2来说,属于直接赋值类型,产生字符串对象放入字符串常量池,s2将会直接从字符串常量池取出已有对象,则s1==s2。s3属于构造方法new一个新对象,需要在堆区开辟一个新的空间,其中value值保存的是字符数组的地址。
上述代码共产生三个对象,一个在常量池中,两个在堆上;产生了4个引用,其中s1和s2指向的是常量池中的对象,s3和s4分别指向的是堆中的对象。
将手动创建的字符串对象置入常量池,并返回置入常量池之后的地址。
看如下的代码:
char[] ch={'a','b','c'};
String s1=new String(ch);
String s2="abc";
System.out.println(s1==s2);
s1:通过new的方式产生的字符串仍然在堆中存储,不会置入常量池
s2:字符串常量产生后直接置入常量池
s1 != s2
char[] ch={'a','b','c'};
String s1=new String(ch);
s1.intern();
String s2="abc";
System.out.println(s1==s2);
此时,将堆上new出来的新对象通过intern()方式手动置入常量池,当“abc”出现时,常量池中已经包含,因此,s2直接返回常量池中存储的地址。
s1 == s2
String s2="abc";
char[] ch={'a','b','c'};
String s1=new String(ch);
s1.intern();
System.out.println(s1==s2);
此时,s2属于直接赋值法将产生的字符串对象直接放入字符串常量池中;
s1属于new出来的对象放置在堆区,s1.intern()方法执行后的返回值无人接收,s1依旧指向的是堆区的字符串对象的地址。
s1 != s2
补充介绍intern()方法
尝试将当前字符串对象置入常量池;
若常量池中不存在该对象内部保存的内容“abc”,则将当前对象置入常量池;
若常量池中已经存在该对象保存的内容,则该方法直接返回常量池中的字符串对象地址!
intern()方法调用后会返回常量池中的字符串地址,调用之后进行接收即可!
如果让s1去接收intern()方法的返回值,此时由于字符串常量池中包含字符串“abc“,因此直接返回常量池中的字符串对象地址,则s1==s2