String、StringBuffer和StringBuilder的区别?String为什么是不可变的?

本文探讨了Java中的String、StringBuffer和StringBuilder的可变性、线程安全性和性能差异。String是不可变的,适合单线程和少量数据;StringBuilder非线程安全但性能优于String;StringBuffer线程安全但效率略低,适用于多线程和大量数据处理。
摘要由CSDN通过智能技术生成

一、可变性

1.String是不可变的

原因:

1.(数组final)保存字符串的数组被 final 修饰且为私有的,并且String 类没有提供/暴露修改这个字符串的方法。

2.(类final)String 类被 final 修饰导致其不能被继承,进而避免了子类破坏 String 不可变。

public final class String implements java.io.Serializable, Comparable<String>, CharSequence {
    /*Java8及以前的版本,使用char数组存储*/
    private final char value[];
    //...
    
    /*Java9及以后的版本,改用byte数组存储*/
    // @Stable 注解表示变量最多被修改一次,称为“稳定的”。
    @Stable
    private final byte[] value;
    //...
}

Java 9 为何要将 String 的底层实现由 char[] 改成了 byte[] ?

新版的 String 其实支持两个编码方案:Latin-1 和 UTF-16。如果字符串中包含的字符没有超过 Latin-1 可表示范围内的字符,那就会使用 Latin-1 作为编码方案。Latin-1 编码方案下byte 占一个字节(8 位),char 占用 2 个字节(16),byte 相较 char 节省一半的内存空间。

JDK 官方就说了绝大部分字符串对象只包含 Latin-1 可表示的字符。

image-20240316152051889

如果字符串中包含的字符超过 Latin-1 可表示范围内的字符,则使用UTF-16编码方案,此时byte 和原本的 char 所占用的空间是一样的。

这是官方的介绍:https://openjdk.java.net/jeps/254

值得注意的是,StringBuffer和StringBuilder对字符串的实现也同步做了改变:

image-20240316152220375

2.StringBuffer和StringBuilder是可变的

原因:StringBuilderStringBuffer 都继承自 AbstractStringBuilder 类,在 AbstractStringBuilder 中也是使用字符数组保存字符串,不过没有使用 finalprivate 关键字修饰,最关键的是这个 AbstractStringBuilder 类还提供了很多修改字符串的方法比如 append 方法。

二、线程安全性

线程安全非线程安全
String、StringBufferStringBuilder

String 中的对象是不可变的,也就可以理解为常量,线程安全

AbstractStringBuilderStringBuilderStringBuffer 的公共父类,定义了一些字符串的基本操作,如 expandCapacityappendinsertindexOf 等公共方法,意味着可以对该对象进行改变,即非线程安全。

StringBuffer 对方法加了同步锁或者对调用的方法加了同步锁,所以是线程安全的。StringBuilder 并没有对方法进行加同步锁,所以是非线程安全的。

三、性能

每次对 String 类型进行改变的时候,都会生成一个新的 String 对象,然后将指针指向新的 String 对象。

StringBuffer 每次都会对 StringBuffer 对象本身进行操作,而不是生成新的对象并改变对象引用。相同情况下使用 StringBuilder 相比使用 StringBuffer 仅能获得 10%~15% 左右的性能提升,但却要冒多线程不安全的风险。

四、使用建议

  1. 操作少量的数据: 适用 String
  2. 单线程操作字符串缓冲区下的大量数据: 适用 StringBuilder
  3. 多线程操作字符串缓冲区下的大量数据: 适用 StringBuffer
  • 43
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值