1、不可变的字符序列(不可变性)
对字符串重新赋值,要重写指定内存区域,不能使用原有的value进行赋值。
对现有的字符串进行拼接操作,也要重新指定内存区域赋值,不能使用原有的value进行赋值。
当调用String的replace方法修改指定字符或者字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
2、通过字面量而不是new给字符串赋值。此字符串声明在字符串常量池中。
3、hashmap的key相当于一个set,不能够有重复。字符串常量池中不会存储内容相同的字符串。
4、String类型的常量池主要使用的方法有两种:
4.1直接使用双引号声明出来的String对象会直接储存在常量池中:String info = "xxx";
4.2如果不是使用双引号声明String对象,可以使用intern()方法。
5、所有字符串都保存在堆当中。永久代比较小,大量字符串会造成内存泄漏,永久代垃圾回收频率低。
6、 字符串拼接:
6.1、s1 = "a"+"b"+"c" (=="abc");s2 = "abc";
====> s1 == s2--->true
====> s1.equals(s2)--->true 常量与常量拼接,结果在字符串常量池.
6.2、 s3 = "abcd"; s4 = "efg";
s5 = "abcdefg";
s6 = s3 + "efg";只要拼接有一个是变量,结果就在堆中.
s7 = s3 + s4;
s6 == s5 ----> false
s5 == s7 ------> false
s6 == s7 ----> false
s8 = s6.intern();
s8 == s5 -----> true
这种情况不算是字符串变量拼接:
final String s1 = "a";
final String s2 = "b";
String s3 = "ab";
String s4 = s1 + s2;
s3 == s4 ----> true
intern():
eg1:
new String("a") 创建了两个对象,一个是new关键字再堆空间创建的;另一个对象是字符串常量池中的对象。
new String("a") +new String("b"):第一个对象是StringBuilder因为出现了变量的拼接操作(“+”);对象二是new,第三个是“a”;第四个new,第五个“b”;第六个在StringBuilder里的toString()(在字符串常量池中没有生成“ab”),new了一个“ab”。
eg2:
1、s是堆空间中对象的地址,s2是字符串常量池中的地址。
2、s3记录的地址为new String(“11”),
jdk6中创建了一个新的对象,也就有新的地址。jdk7常量池中存放的不是“11”,而是指向在堆空间中new出来的对象。
eg3:
总结:
eg4:
堆空间内指向常量池ab的地址,栈引用的是堆空间的地址。s1指向是堆中的,s2指向是常量池里的。
7、 大型网站需要内存中存储大量的字符串,这时候如果字符串都调用intern()方法,会明显降低内存的大小。