String基本特性
- String字符串,使用一对“”引起来表示
- String s1 = “lotus”; //字面量定义方式
- String s2 = new String(“hello”)
- String声明为final,不可被继承
- String实现了Serializable接口,表示字符串支持序列化,实现Comparable接口,表示String可以比较大小
- String在JDK8以前内部定义了final char[] value用于存储字符串数据,jdk9改为byte[]
- String代表不可变字符序列,称不可变性
- 当对字符串重新赋值时,重写指定内存区域赋值,不能使用原有的value进行赋值
- 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值
- 当调用String的replace()方法修改指定字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值
- 通过字面量的方式(区别于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是可设置的最小值
String内存分配
- Java语言中有8种基本数据类型和一种比较特殊的类型String,这些类型为了使它们运行过程中速度更快,更节省内存,都提供了一种常量池的概念
- 常量池类似于一个Java系统级别提供的缓存。8种基本数据类型的常量池都是系统协调的,String的常量池比较特殊,主要使用方法有两种
- 直接使用双引号声明出的String对象会直接存储在常量池中
- 不使用双引号声明的String对象,可以使用String提供的intern()方法
- Java6及以前,字符串常量池存放在永久代
- Java7中对字符串池的逻辑做了改变,将字符串常量池位置调整到Java堆内
- 所有的字符串都保存在堆中,和其他普通对象一样,这样可以让你在进行调优应用时仅需调整堆大小就可以了
- 字符串常量池概念原本使用得比较多,但是这个改动使我们有足够的理由重新考虑使用String.intern()
- Java8元空间,字符串常量池在堆中
字符串拼接操作
@Test
public void test1() {
String s1 = "a" + "b" + "c";
String s2 = "abc";
System.out.println(s1 == s2);
System.out.println(s1.equals(s2));
}
true
true
- 常量池中不会存在相同内容的常量
- 只要其中有一个是变量,结果就在堆中,变量拼接原理是StringBuilder
@Test
public void test2() {
String s1 = "javaEE";
String s2 = "hadoop";
String s3 = "javaEEhadoop";