String、StringBuilder、StringBuffer
String
真正不可变有下面几点原因:
- 保存字符串的数组被
final
修饰且为私有的,并且String
类没有提供/暴露修改这个字符串的方法。 String
类被final
修饰导致其不能被继承,进而避免了子类破坏String
不可变。
String:不可变,线程安全
StringBuilder:可变,单线程,线程不安全
StringBuffer:可变,多线程,线程安全(方法加了同步锁)
String可以通过反射改变值:
public class Main { public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException, NoSuchFieldException { String s = "abc"; Class clz = s.getClass(); //需要使用getDeclaredField(), getField()只能获取公共成员字段 Field field = clz.getDeclaredField("value"); field.setAccessible(true); char[] ch =(char[])field.get(s); ch[1] = '8'; //field.set(s,ch); System.out.println(s);//a8c } }
字符串拼接用“+” 还是 StringBuilder?
字符串对象通过“+”的字符串拼接方式,实际上是通过 StringBuilder
调用 append()
方法实现的,拼接完成之后调用 toString()
得到一个 String
对象 。
不过,在循环内使用“+”进行字符串的拼接的话,存在比较明显的缺陷:编译器不会创建单个 StringBuilder
以复用,会导致创建过多的 StringBuilder
对象。
String s1 = new String("abc");这句话创建了几个字符串对象?
会创建 1 或 2 个字符串对象。
(1)如果字符串常量池中不存在字符串对象“abc”的引用,那么它将首先在字符串常量池中创建,然后在堆空间中创建,因此将创建总共 2 个字符串对象。
(2)如果字符串常量池中已存在字符串对象“abc”的引用,则只会在堆中创建 1 个字符串对象“abc”。