- String 类型是不可变字符串序列,线程安全。字符串字面值属于常量,在常量池中存储。(jdk8中String类中字符串是一个final修饰的字符数组
private final char value[];
,所以不可变。
final 修饰的变量一旦初始化,就不能被修改,如果是类变量,只能在构造方法中初始化,在其他方法中如果初始化,编译器会报错,包括String. 但是也有例外.如Map,List等,即使你用final修饰了,依然可以put和add 说到底,还是引用问题.final意味着引用不可变罢了, 反射可以改变对象中final修饰的变量)(String类是不可变,所以是线程安全的)- StringBuffer:可变字符串序列,长度可变。线程安全(synchronized),效率低;
- StringBuilder:可变字符串序列,线程不安全,效率高
注意:
jdk8中,+
其实本质上就是使用StringBuilder。
即:
String a = "a";
// 加
String b = a + "b";
等价于:
StringBuilder asb = new StringBuilder(a);
asb.append("b");
String bsb = asb.toString();
所以,每次+
。都会创建StringBuilder
对象。
那么建议:
- for循环中拼接不用
+
, 因为每次都会创建一个StringBuilder
对象 - 普通的拼接直接用
+
,因为底层也是StringBuilder
,效率一样,可读性更高。
不可变类都是线程安全的,线程安全的不一定是不可变类。
StringBuffer是可变类,用synchronize锁来保证线程安全
StringBuffer 和 StringBuilder 继承了 AbstractStringBuilder
但是StringBuffer 把 父类的方法 都加了synchronize。所以StringBuffer是线程安全的,但是效率低。
注意:
- String 是final修饰的 char[] 数组,所以不可变
- 而StringBuffer和StringBuilder的父类AbstractStringBuilder的char[] 并没有用final修饰。
AbstractStringBuilder的append(str) 方法,实际上,是调用了String的getChars() 方法,把str拼接到char[] 数组上,然后添加这个StringBuilder的长度属性。
public void getChars(int srcBegin, int srcEnd, char[] dst, int dstBegin)
-
srcBegin – 字符串中要复制的第一个字符的索引。
-
srcEnd – 字符串中要复制的最后一个字符之后的索引。
-
dst – 目标数组。
-
dstBegin – 目标数组中的起始偏移量x
什么是不可变类?
实现不可变类的步骤
1)类声明为final,不可以被继承
2)所有成员是私有的,不允许直接被访问
3)对变量不要setter方法
4)所有可变的变量是final的,只能赋值一次
5)通过构造器初始化所有成员,进行深拷贝
6)在getter方法中不能返回对象本身,返回对象的拷贝
String a = "abc";
a = "bcd";
System.out.println(a);
注意:变得只是a的引用,并不是字符串本身,字符串在字符串常量池中
字符串常量池
java9
Java9改进了字符串(包括String、StringBuffer、StringBuilder)的实现。在Java9以前字符串采用char[]数组来保存字符,因此字符串的每个字符占2字节;而Java9的字符串采用byte[]数组再加一个encoding-flag字段来保存字符,因此字符串的每个字符只占1字节。所以Java9的字符串更加节省空间,字符串的功能方法也没有受到影响。