String 和StringBuffer,StringBuilder 的区别是什么?String 为什么是不可变的?
while ((line = br.readLine()) != null) {
if (StringUtils.isNotEmpty(buf.toString())) {
buf.append("\r\n");
}
buf.append(line);
}
jstack打印出来线程栈信息,能看到线程栈目前运行在 if (StringUtils.isNotEmpty(buf.toString())) ,占了大量内存
在对于循环操作字符串操作不要使用string类型,像上段代码会导致线上服务器负载过高发生了报警,原因是因为string类型每次对 String 类型进行改变的时候,都会生成一个新的 String 对象,然 后将指针指向新的 String 对象。
public String(char value[], int offset, int count) {
if (offset < 0) {
throw new StringIndexOutOfBoundsException(offset);
}
if (count < 0) {
throw new StringIndexOutOfBoundsException(count);
}
// Note: offset or count might be near -1>>>1.
if (offset > value.length - count) {
throw new StringIndexOutOfBoundsException(offset + count);
}
this.offset = 0;
this.count = count;
this.value = Arrays.copyOfRange(value, offset, offset+count);
}`
string类型的构造函数的最后一行的value并不是指向型的,而是使用系统拷贝函数进行内存复制。
每次对 String 类型进行改变的时候,都会生成一个新的 String 对象,然 后将指针指向新的 String 对象。 StringBuffer 每次都会对 StringBuffer 对象本身进行操作,而不是生成新 的对象并改变对象引用。相同情况下使用StirngBuilder 相比使用 StringBuffer 仅能获得 10%~15% 左右的性能提升,但却要冒多线程不安全的风险。
线程安全性
String 中的对象是不可变的,也就可以理解为常量,线程安全。 AbstractStringBuilder 是 StringBuilder 与 StringBuffer 的公共父类,定义 了一些字符串的基本操作,如 expandCapacity.append.insert.indexOf 等公 共 方法。StringBuffer 对方法加了同步锁或者对调用的方法加了同步锁,所以 是线程安全的。StringBuilder 并没有对方法进行加同步锁,所以是非线程安全 的。
对于三者使用的总结: 1. 操作少量的数据 => 使用 String 2. 单线程操作字符串缓冲区下操作大量数据 => 使用 StringBuilder 3. 多线程操作字符串缓冲区下操作大量数据 => 使用 StringBuffer
本文详细介绍了Java中String、StringBuffer和StringBuilder的区别。String是不可变对象,每次修改都会创建新对象,导致内存消耗。而StringBuffer和StringBuilder在循环操作字符串时更高效,尤其StringBuilder在单线程环境下。StringBuffer是线程安全的,适合多线程环境,而StringBuilder则无此保障。因此,根据操作频率和线程安全需求选择合适的字符串处理类至关重要。
465

被折叠的 条评论
为什么被折叠?



