String:字符串,使用一对""引起了表示。
String s1 = “zmh”;//字面量的定义方式
String s2 = new String(“hello”);
String声明为final,不可被继承
String实现了Serializable接口:表示字符串是支持序列化的。
实现了Comparable接口:表示String可比较大小
String在jdk8及以前内部定义了final char[] value用于存储字符串数据。jdk9时改为final byte[] value。使用byte数组可以减少一半的内存,byte使用一个字节来存储一个char字符,char使用两个字节来存储一个char字符。
String:代表不可变的字符序列。简称:不可变性。
当对字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值。
@Test
public void test01(){
String s1 = "abc";//字母量定义的方式,“abc”存储在字符串常量池中
String s2 = "abc";
s1 = "hello";//这个地方修改了,不是在原有的基础修改,而是重新指向hello,不会影响s2的值
System.out.println(s1 == s2);// false
System.out.println(s1);
System.out.println(s2);
}
当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
@Test
public void test02(){
String s1 = "abc";
String s2 = "abc";
s2 += "def";// 不能在现有"abc"的基础直接添加"def",因为定义好了一个String,底层的数组长度就固定了。而且在字符串常量池中做修改就要重新创建,体现不可变性
System.out.println(s2);
System.out.println(s1);
}
当调用String的replace()方法修改指定字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
@Test
public void test03(){
String s1 = "abc";
String s2 = s1.replace('a','m');//如果能直接在原值上修改,那么s1和s2都变成了mbc。但不是这样的
System.out.println(s1);
System.out.println(s2);
}
通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。
字符串常量池中是不会存储相同内容的字符串的
·因为String的String Pool是一个固定大小的HashTable,默认值长度是1009。如果放进String Pool的String非常多,就会造成Hash冲突严重,从而导致链表会很长,而链表长了后直接会造成的影响就是当调用String.intern时性能会大幅下降。
·使用-XX:StringTableSize可设置StringTable的长度。
·在jdk6中StringTable是固定的,就是1009的长度,所以如果常量池中的字符串过多就会导致效率下降很快。StringTableSize设置没有要求。
·在jdk7中,StringTable的长度默认值是60013,StringTableSize设置没有要求。
·jdk8开始,设置StringTable的长度的话,1009是可设置的最小值。
jdk6中:
jdk7中: