一、概述
- String类声明为final的,不可被继承
- String实现了Serializable接口:表示字符串是支持序列化的;实现了Comparable接口:表示String可以比较大小
- 底层存储:
- JDK8及以前底层使用final char[]数组
- JDK9及以后改用final byte[]数组,是为了节省空间
- Java中的String具有不可变性
- 如下操作需要生成新的字符串,而不是修改原有的字符串。
- 对现有的字符进行连接操作时
- 对字符串重新赋值时
- 调用String的replace()方法修改指定字符或字符串时
- 如何保证String的不可变性
- char[]使用final修饰->无法指向新的char[]数组
- String内部没有提供修改(setXX)的方法->内容不会变
- 为什么将String设计为不可变?
- 字符串在实际的开发中使用太频繁,为了提高执行效率,把字符串放到了方法区的“字符串常量池”当中
- 不可变字符串,使得编译器可以让字符串共享
- 如下操作需要生成新的字符串,而不是修改原有的字符串。
- 字符串常量池
- 常量池中不会存在相同内容的常量
- 通过字符串字面量赋值时,数据是在常量池中
- String Pool是一个固定大小的Hashtable(不会扩容)
- 数组+链表的哈希表(拉链法解决hash冲突)
- 使用- XX:StringTableSize可设置StringTable的长度
二、字符串拼接操作
- 常量与常量的拼接结果在常量池(原理是编译期优化)
- 只要其中有一个是变量,结果就在堆中(变量拼接的原理是StringBuilder)
- 如果拼接的结果调用intern()方法,则主动将常量池中还没有的字符串对象放入池中,并返回对象地址
三、intern()方法
属于native()方法
作用:将String添加到字符串常量池中
执行过程:如果当前字符串在字符串常量池中已存在,则返回其在常量池中的地址即可;若不存在,则会将当前字符串添加到字符串常量池后,并返回其地址(JDK6和JDK7以后有差别)
Interned String就是确保字符串在内存里只有一份拷贝(节省空间,提升速度)
tips:当需要在内存中存储大量的字符串时,调用intern()方法,会明显降低内存的大小
四、如何保证变量s指向的是字符串常量池中的数据?
方法1:使用字符字面量
方法2:使用intern()方法