一、String
创建String对象有两种形式,一种是直接赋值,如String s = "hello","hello"存储在字符串常量池; 另一种是通过new String()对象,通过调用构造方法创建对象,如new String({'a','b','c'}) 或者 new String({97,98,99}),此时是在堆上创建一个对象。当直接赋值给变量时,会先在常量池中判断该字符串是否存在,如果存在则直接赋值给新变量(两个变量的引用相同),如果不存在,则在常量池中创建一个新的字符串,然后将引用赋值给变量(两个变量的引用不同)。而通过new对象的形式,不会在常量池中查找,变量一定是指向新的引用。
String可以通过 + 拼接字符串,但这种拼接字符串的方式,会创建很多临时字符串对象,效率较低。以下代码会在常量池中创建 s1、s2、s3、"aaabbb"(s1+s2) 、"aaabbbccc"(s1+s2+s3)五个对象,因为常量池中的字符串不可变(private final修饰),每次拼接,都会创建一个新的字符串。因此,在大量的字符串拼接操作时,通常使用StringBuilder对象的append方法,而且通过反编译可以看出,在使用 + 拼接字符串时,jvm是调用StringBuilder对象的append方法,即创建"aaa"和一个空的StringBuilder对象,然后StringBuilder对象append("aaa"),然后再append其他字符串,最后通过StringBuilder的toString方法,将StringBuilder对象转为字符串。
String s1 = "aaa";
String s2 = "bbb";
String s3 = "ccc";
String s = s1 + s2 + s3
二、StringBuild
StringBuilder继承于AbstractStringBuilder,底层使用数组接收字符串。在字符串拼接时,StringBuilder不会创建新的对象,而是在数组后面填充。StringBuilder通过重写的toString()方法,可以将StringBuilder对象转为String对象。StringBuilder的很多方法返回值为this,即不会创建新的对象。
三、StringBuffer
StringBuffer也是继承于AbstractStringBuilder,底层也使用数组接收字符串,但StringBuffer的很多方法有synchronized修饰,保证了线程安全,但加锁操作也有性能开销。为了节省开销,才有了StringBuilder。
总结:String、StringBuilder、StringBuffer底层接收字符串都是基于数组实现。String由于 private final修饰,所以字符串对象不可变,在对字符串进行拼接、截取等操作时,会创建很多新的字符串对象,效率不高,但也正是因为String对象的不可变,String对象可以作为哈希的键使用,也具有安全性;StringBuilder和StringBuffer都继承于AbstractStringBuilder,都是针对字符串操作的类,但StringBuffer有synchronized修饰,可以保证线程安全。
补充:在对字符串拼接操作时,也可以使用StringJoiner