StringBuilder

上一节,我们学习了 Java 的 String 类,并介绍了其常用方法。本小节我们来介绍字符串的另外一个类:StringBuilder,我们将会了解到 StringBuilderString差异StringBuilder使用场景,也会介绍与 StringBuilder 类对应的 StringBuffer 类,StringBuilder使用方法以及其常用方法是本小节的重点学习内容。

1. StringBuilder 概述

1.1 什么是 StringBuilder

String 相似,StringBuilder 也是一个与字符串相关的类,Java 官方文档给 StringBuilder 的定义是:可变的字符序列。

1.2 为什么需要 StringBuilder

在 Java 字符串的学习中,我们知道了字符串具有不可变性,当频繁操作字符串时候,会在常量池中产生很多无用的数据(回忆图示)。

StringBuilderString 不同,它具有可变性。相较 String 类不会产生大量无用数据,性能上会大大提高。

因此对于需要频繁操作字符串的场景,建议使用 Stringbuilder 类来代替 String 类。

2. StringBuffer 概述

2.1 定义

了解了 StringBuilder 类 ,StringBuffer 也是不得不提的一个类,Java 官方文档给出的定义是:线程安全的可变字符序列。

2.2 与前者的区别

StringBufferStringBuilder 的前身,在早期的 Java 版本中应用非常广泛,它是 StringBuilder 的线程安全版本(线程我们将在后面的小节中介绍),但实现线程安全的代价是执行效率的下降

你可以对比 StringBuilderStringBuffer接口文档,它们的接口基本上完全一致。为了提升我们代码的执行效率,在如今的实际开发中 StringBuffer 并不常用。因此本小节的重点在 StringBuilder 的学习。

3. StringBuilder 的常用方法

3.1 构造方法

StringBuilder 类提供了如下 4 个构造方法:

  1. StringBuilder() 构造一个空字符串生成器,初始容量为 16 个字符;
  2. StringBuilder(int catpacity) 构造一个空字符串生成器,初始容量由参数 capacity 指定;
  3. StringBuilder(CharSequence seq) 构造一个字符串生成器,该生成器包含与指定的 CharSequence 相同的字符。;
  4. StringBuilder(String str) 构造初始化为指定字符串内容的字符串生成器。

其中第 4 个构造方法最为常用,我们可以使用 StringBuilder 这样初始化一个内容为 hello 的字符串:

StringBuilder str = new StringBuilder("Hello");

3.2 成员方法

StringBuilder 类下面也提供了很多与 String 类相似的成员方法,以方便我们对字符串进行操作。下面我们将举例介绍一些常用的成员方法。

3.2.1 字符串连接

可以使用 StringBuilderStringBuilder append(String str) 方法来实现字符串的连接操作。

我们知道,String 的连接操作是通过 + 操作符完成连接的:

String str1 = "Hello";
String str2 = "World";
String str3 = str1 + " " + str2;

如下是通过 StringBuilder 实现的字符串连接示例:

实例演示
预览 复制
复制成功!
public class ConnectString1 {
    public static void main(String[] args) {
        // 初始化一个内容为 Hello 的字符串生成器
        StringBuilder str = new StringBuilder("Hello");
        // 调用append()方法进行字符串的连接
        str.append(" ");
        str.append("World");
       	System.out.println(str);
    }
}
运行案例 点击 "运行案例" 可查看在线运行效果

运行结果:

Hello World

由于 append() 方法返回的是一个 StringBuilder 类型,我们可以实现链式调用。例如,上述连续两个 append() 方法的调用语句,可以简化为一行语句:

str.append(" ").append("World");

如果你使用 IDE 编写如上连接字符串的代码,可能会有下面这样的提示(IntelliJ idea 的代码截图):

提示内容说可以将 StringBuilder 类型可以替换为 String 类型,也就是说可以将上边地代码改为:

String str = "Hello" + " " + "World";

这样写并不会导致执行效率的下降,这是因为 Java 编译器在编译和运行期间会自动将字符串连接操作转换为 StringBuilder 操作或者数组复制,间接地优化了由于 String 的不可变性引发的性能问题。

值得注意的是,append() 的重载方法有很多,可以实现各种类型的连接操作。例如我们可以连接 char 类型以及 float 类型,实例如下:

实例演示
预览 复制
复制成功!
public class ConnectString2 {
    public static void main(String[] args) {
        StringBuilder str = new StringBuilder("小明的身高为");
        str.append(':').append(172.5f);
        System.out.println(str);
    }
}
运行案例 点击 "运行案例" 可查看在线运行效果

运行结果:

小明的身高为:172.5

上面代码里连续的两个 append() 方法分别调用的是重载方法 StringBuilder append(char c)StringBuilder append(float f)

3.2.2 获取容量

可以使用 int capacity() 方法来获取当前容量,容量指定是可以存储的字符数(包含已写入字符),超过此数将进行自动分配。注意,容量与长度(length)不同,长度指的是已经写入字符的长度。

例如,构造方法 StringBuilder() 构造一个空字符串生成器,初始容量为 16 个字符。我们可以获取并打印它的容量,实例如下:

实例演示
预览 复制
复制成功!
public class GetCapacity {
    public static void main(String[] args) {
        // 调用StringBuilder的无参构造方法,生成一个str对象
        StringBuilder str = new StringBuilder();
        System.out.println("str的初始容量为:" + str.capacity());
        // 循环执行连接操作
        for (int i = 0; i < 16; i ++) {
            str.append(i);
        }
        System.out.println("连接操作后,str的容量为" + str.capacity());
    }
}
运行案例 点击 "运行案例" 可查看在线运行效果

运行结果:

str的初始容量为:16
连接操作后,str的容量为34

3.2.3 字符串替换

可以使用 StringBuilder replace(int start, int end, String str) 方法,来用指定字符串替换从索引位置 start 开始到 end 索引位置结束(不包含 end)的子串。实例如下:

实例演示
预览 复制
复制成功!
public class StringReplace {
    public static void main(String[] args) {
        // 初始化一个内容为 Hello 的字符串生成器
        StringBuilder str = new StringBuilder("Hello World!");
        // 调用字符串替换方法,将 World 替换为 Java
        str.replace(6, 11, "Java");
        // 打印替换后的字符串
        System.out.println(str);
    }
}
运行案例 点击 "运行案例" 可查看在线运行效果

运行结果:

Hello Java!

也可使用 StringBuilder delete(int start, int end) 方法,先来删除索引位置 start 开始到 end 索引位置(不包含 end)的子串,再使用 StringBuilder insert(int offset, String str) 方法,将字符串插入到序列的 offset 索引位置。同样可以实现字符串的替换,例如:

StringBuilder str = new StringBuilder("Hello World!");
str.delete(6, 11);
str.insert(6, "Java");

3.2.4 字符串截取

可以使用 StringBuilder substring(int start) 方法来进行字符串截取,例如,我们想截取字符串的后三个字符,实例如下:

实例演示
预览 复制
复制成功!
public class StringSub {
    public static void main(String[] args) {
        StringBuilder str = new StringBuilder("你好,欢迎来到慕课网");
        String substring = str.substring(7);
        System.out.println("str截取后子串为:" + substring);
    }
}
运行案例 点击 "运行案例" 可查看在线运行效果

运行结果:

str截取后子串为:慕课网

如果我们想截取示例中的” 欢迎 “二字,可以使用重载方法 StringBuilder substring(int start, int end) 进行截取:

String substring = str.substring(3, 5);

3.2.5 字符串反转

可以使用 StringBuildr reverse() 方法,对字符串进行反转操作,例如:

实例演示
预览 复制
复制成功!
public class StringReverse {
    public static void main(String[] args) {
        StringBuilder str = new StringBuilder("Hello Java");
        System.out.println("str经过反转操作后为:" + str.reverse());
    }
}
运行案例 点击 "运行案例" 可查看在线运行效果

运行结果:

str经过反转操作后为:avaJ olleH

4. 小结

本小节我们介绍了 Java 的 StringBuilder 类,它具有可变性,对于频繁操作字符串的场景,使用它来代替 String 类可以提高程序的执行效率;也知道了 StringBufferStringBuilder 的线程安全版本,官方更推荐使用 StringBuilder;最后我们介绍了 StringBuilder 的常用构造方法和成员方法,如果你想了解更多关于 StringBuilder 的接口,可以翻阅官方文档进行学习。