JAVA面试:String、StringBuffer和StringBuilder区别

简介: `String`是不可变的,`StringBuffer`和`StringBuilder`是可变的。`String`的不可变性源于其内部的`final char[]`数组,这意味着每次修改都会创建新对象。`StringBuffer`线程安全,方法同步,适合多线程环境,但效率较低;`StringBuilder`非线程安全,无同步,单线程中效率更高。两者初始容量相同,扩容机制也一样。

 


1. String、StringBuffer和StringBuilder区别

答:String是不可变的,而StringBuffer和StringBuilder是可变的。

这样回答肯定是不满意的,面试官希望听到的是你能说出个为什么?而不是给我一个结论,只说结论面试官会认为你只是背了面试题而已。

2. 为什么String是不可变的?

简单的讲String是内部结构如下

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];
...
}

image.gif

是用final char value[]来存储字符串, 所以上不可变的。这么说肯定凉了。

final说明: 修饰的类不能被继承、修饰的方法不能被重写、修饰的基本数据类型变量是其值不能改变、修饰的引用类型变量是不能再指向其他对象。

所以final修饰的char value[],其组据是可变的。

为什么是不可变呢? 先看String的Api,提供了哪些public方法

image.png image.gif image.png

从源码中可以看出

  • String 并没有提供一个可以修改数组值的方法
  • String是final类,不能被继承,也不可能有子类来修改

3.为什么StringBuffer和StringBuilder是可变的?

StringBugger和StringBuilder都继承自AbstractStringBuilder,
StringBugger和StringBuilder存放字符串都放在AbstractStringBuilder的char[] value中。
而AbstractStringBuilder提供如append来修改value的值。所以是可变的。

image.gif

4. StringBuffer和StringBuilder的区别?

1. 先看看源码,如下

image.png

StringBuffer的公开方法加了对象锁,所以是线程安全的,而StringBuilder没有,所以是线程不安全

2. StringBuffer的API加了对象锁,执行速度慢(效率慢)。StringBuilder未加锁,执行速度快(效率高)。

3. toString方法不同,直接看源码

image.png

注: toStringCache的定义是 private transient char[] toStringCache

从源码看出:

  • StringBuffer的toString都会直接使用缓存区的toStringCache值来构造一个字符串
  • 而StringBuilder是需要复制一次字符数组,再构造一个字符串

5. StringBuffer和StringBuilder相同之处?

  1. 字符串都可变
  2. 默认初始容量都是16
  3. 扩容机制相同。源码如下:
都是调父类AbstractStringBuilder 的扩容方法,源码如下:
    private void ensureCapacityInternal(int minimumCapacity) {
        // overflow-conscious code
        if (minimumCapacity - value.length > 0) {
            value = Arrays.copyOf(value,
                    newCapacity(minimumCapacity));
        }
    }
    private int newCapacity(int minCapacity) {
        // overflow-conscious code
        int newCapacity = (value.length << 1) + 2;
        if (newCapacity - minCapacity < 0) {
            newCapacity = minCapacity;
        }
        return (newCapacity <= 0 || MAX_ARRAY_SIZE - newCapacity < 0)
            ? hugeCapacity(minCapacity)
            : newCapacity;
    }
    private int hugeCapacity(int minCapacity) {
        if (Integer.MAX_VALUE - minCapacity < 0) { // overflow
            throw new OutOfMemoryError();
        }
        return (minCapacity > MAX_ARRAY_SIZE)
            ? minCapacity : MAX_ARRAY_SIZE;
    }

image.gif


相关文章
|
6月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
392 5
|
8月前
|
存储 SQL 缓存
Java字符串处理:String、StringBuilder与StringBuffer
本文深入解析Java中String、StringBuilder和StringBuffer的核心区别与使用场景。涵盖字符串不可变性、常量池、intern方法、可变字符串构建器的扩容机制及线程安全实现。通过性能测试对比三者差异,并提供最佳实践与高频面试问题解析,助你掌握Java字符串处理精髓。
|
9月前
|
自然语言处理 Java Apache
在Java中将String字符串转换为算术表达式并计算
具体的实现逻辑需要填写在 `Tokenizer`和 `ExpressionParser`类中,这里只提供了大概的框架。在实际实现时 `Tokenizer`应该提供分词逻辑,把输入的字符串转换成Token序列。而 `ExpressionParser`应当通过递归下降的方式依次解析
458 14
|
6月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
311 1
|
6月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
319 1
|
7月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
299 0
|
7月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
480 16
下一篇
开通oss服务