1. 基本特性
- String类是final的,不可被继承
- String具有不可变性
- String实现了Serializable接口,支持序列化的;实现了Comparable接口,可以比较大小
- JDK8及以前,使用final char[] value存储字符串数据,JDK9时改为byte[]
-
- 原因:大多数的String字符串只包含拉丁文字符,这些字符只需要一个byte的空间进行存储,如果用一个char来存储一个拉丁字符,由于一个char占了两个空间,那么就会有一半的空间被浪费
- 用byte[]加上编码标记(在数组的最后是编码标记:utf8、gbk)进行存储,节约了一些空间
-
- StringBuilder和StringBuffer也做了同样的改动
2. 字符串常量池
字符串常量池:String Pool,是一个固定大小的HashTable,实现此功能的是StringTable类
- StringTable在每个HotSpot VM的实例只有一份,被所有的类共享
- JDK6中,StringTable的默认长度是1009
-
- 如果放入String Pool中的String非常多,就会造成hash冲突,导致链表过长
- JDK7开始,默认长度是60013,JDK8开始,1009是可设置的最小值
-
-
- 可以使用-XX:StringTableSize设置StringTable的长度,比如:-XX:StringTableSize=66666
-
- JDK7开始,字符串常量放在堆中,之前在方法区
-
- 原因:
-
-
- 方法区默认容量小,不适合存储大量字符串
- 垃圾回收的频率低
-
java中的字符串都在字符串常量池中,创建一个字符串时,先看常量池中是否存在
- 是:直接返回字符串常量池中该对象的地址
- 否:创建字符串放入常量池中,然后返回这个字符串的地址
new String()
String str ="dd":可能创建一个或者不创建对象
如果”dd”在常量池中不存在,会创建一个String对象”dd”,然后str指向这个内存地址
代码演示
public static void main(String[] args) throws InterruptedException { System.out.println(); System.out.println("daji"); System.out.println("yase"); System.out.println("daji"); System.out.println("yase"); }
Debug运行后
上图可以发现:执行最后两行的时候,并没有新增字符串
String str = new String(“dd”):至少会创建一个对象,也有可能创建两个。
new关键字,肯定会在堆中创建一个对象,如果”dd”在常量池中不存在,会创建一个String对象”dd”
代码演示:
public static void main(String[] args) throws InterruptedException { System.out.println(); System.out.println("daji"); System.out.println("yase"); String str = new String("lvbu");//创建两个String对象 String str2 = new String("daji");//创建一个String对象 }