本来String类已经能够完成字符串操作的所有功能,为何Java又提供了专门的StringBuffer和StringBuilder呢?这要从String类的设计说起了,查看String的源码,发现其内部采用字符数组保存字符串,如下所示:

    private final char value[];

可是问题在于,这个字符数组被final修饰了,意味着数组大小不可改变。若想将现有字符串拼接一段字符形成新串(无论是加号拼接还是调用format方法),String类就无法扩充现有的字符数组,只能重新生成新的String对象。然而在需要频繁拼接字符串的场合,不断地重新生成String对象,这涉及到内存数据的反复销毁和反复分配,无疑对程序性能产生巨大的影响。
为了解决频繁拼接带来的性能问题,一开始Java提供了StringBuffer类,其内部的value数组是大小可变的,在拼接字符串时无需重新生成StringBuffer对象,由此避免了对象创建与对象销毁带来的开销。StringBuffer的拼接方法名叫append,因为该方法同时也返回拼接后的StringBuffer对象,所以拼接多个字符串时只需连续调用append即可。
不过Java把StringBuffer类设计成线程安全,它的append方法被synchronized修饰,表示这是个同步方法。既然是同步方法,那么调用之时就会先加锁,等调用结束再解锁。该设计的意图是让StringBuffer类更加健壮,但是拼接字符串几乎不存在多线程并发的场景,绝大多数情况都是按顺序拼接,谁会脑袋抽筋中途插队拼接呢?于是这个锁同步设计形同鸡肋。
鉴于加解锁也会带来额外的性能损耗,因此JDK1.5又增加了StringBuilder类,该类的实现方式与StringBuffer类大同小异,主要区别在于StringBuilder类的append方法去掉了synchronized修饰,从而省去了无谓的加解锁操作。
下面通过实际代码演示一下StringBuffer、StringBuilder和String三者在频繁拼接时候的性能表现,每种对象分别拼接字符串199999次:

    public static void main(String[] argv) {
testBuffer(199999);
testBuilder(199999);
testString(199999);
} // 测试StringBuffer的字符串拼接
private static void testBuffer(int count) {
long beginTime = System.currentTimeMillis();
StringBuffer buffer = new StringBuffer("");
for (int i=0; i<=count; i++) {
buffer.append("a");
}
long endTime = System.currentTimeMillis();
String costTime = String.format("%.3fs", (endTime-beginTime)/1000.0);
System.out.println("testBuffer cost "+costTime);
} // 测试StringBuilder的字符串拼接
private static void testBuilder(int count) {
long beginTime = System.currentTimeMillis();
StringBuilder builder = new StringBuilder("");
for (int i=0; i<=count; i++) {
builder.append("a");
}
long endTime = System.currentTimeMillis();
String costTime = String.format("%.3fs", (endTime-beginTime)/1000.0);
System.out.println("testBuilder cost "+costTime);
} // 测试String的字符串拼接
private static void testString(int count) {
long beginTime = System.currentTimeMillis();
String str = "";
for (int i=0; i<=count; i++) {
str += "a";
}
long endTime = System.currentTimeMillis();
String costTime = String.format("%.3fs", (endTime-beginTime)/1000.0);
System.out.println("testString cost "+costTime);
}

运行测试代码,观察到如下所示的结果日志:

testBuffer cost 0.022s
testBuilder cost 0.010s
testString cost 13.865s

可见StringBuilder耗时最短,StringBuffer其次当然差距也不大,String耗时最长并且差距显著增大。上述案例说明,在大量拼接字符串的场合,采取StringBuilder实现是最经济的。


更多Java技术文章参见《Java开发笔记(序)章节目录

Java开发笔记(一百五十四)StringBuffer和StringBuilder的由来的更多相关文章

  1. Java开发笔记(五十四)内部类和嵌套类

    通常情况下,一个Java代码文件只定义一个类,即使两个类是父类与子类的关系,也要把它们拆成两个代码文件分别定义.可是有些事物相互之间密切联系,又不同于父子类的继承关系,比如一棵树会开很多花朵,这些花儿 ...

  2. Java开发笔记(三十四)字符串的赋值及类型转换

    不管是基本的char字符型,还是包装字符类型Character,它们的每个变量只能存放一个字符,无法满足对一串字符的加工.为了能够直接操作一连串的字符,Java设计了专门的字符串类型String,该类 ...

  3. Java开发笔记(五十)几种开放性修饰符

    前面介绍子类继承父类的时候,提到了public(公共)和private(私有)两个修饰符,其中public表示它所修饰的实体是允许外部访问的:而private表示它所修饰的实体不允许外部访问,只能在当 ...

  4. Java开发笔记(五十六)利用枚举类型实现高级常量

    前面介绍了联合利用final和static可实现常量的定义,该方式用于简单的常量倒还凑合,要是用于复杂的.安全性高的常量,那就力不从心了.例如以下几种情况,final结合static的方式便缺乏应对之 ...

  5. Java开发笔记(八十四)文件与目录的管理

    程序除了处理内存中的数据结构,还要操作磁盘上的各类文件,这里的磁盘是个统称,泛指可以持久保留数据的存储介质,包括但不限于:插在软驱中的软盘.固定在机箱中的硬盘.插在光驱中的光盘.插在USB接口上的U盘 ...

  6. Java开发笔记(二十四)方法的组成形式

    经过前面的学习,我们发现演示的Java代码越来越复杂,而且每个例子的代码都堆在入口方法main内部,这会导致如下问题:1.一个方法内部堆砌了太多的代码行,看着费神,维护起来也吃力:2.部分代码描述的是 ...

  7. Java开发笔记(五十二)对象的类型检查

    前面介绍了类的多态性,来自于鸡类的实例chicken,既能用来表达公鸡实例,也能用来表达母鸡实例.可是这导致了一个问题,假如在call方法内部需要手工判断输入参数属于公鸡实例还是母鸡实例,那该如何是好 ...

  8. Java开发笔记(五十五)关键字static的用法

    前面介绍嵌套类的时候讲到了关键字static,用static修饰类,该类就变成了嵌套类.从嵌套类的用法可知,其它地方访问嵌套类之时,无需动态创建外层类的实例,直接创建嵌套类的实例就行.其实static ...

  9. Java开发笔记(五十八)简单接口及其实现

    前面介绍了抽象方法及抽象类的用法,看似解决了不确定行为的方法定义,既然叫唤动作允许声明为抽象方法,那么飞翔.游泳也能声明为抽象方法,并且鸡类涵盖的物种不够多,最好把这些行为动作扩展到鸟类这个群体,于是 ...

  10. Java开发笔记(五十九)Java8之后的扩展接口

    前面介绍了接口的基本用法,有心的朋友可能注意到这么一句话“在Java8以前,接口内部的所有方法都必须是抽象方法”,如此说来,在Java8之后,接口的内部方法也可能不是抽象方法了吗?之所以Java8对接 ...

随机推荐

  1. KGDB调试Linux内核与模块

    前言 内核 5.10 版本 openEuler 使用 yum install 下载了源码,并且通过两个 VMware 虚拟机进行调试 ubuntu 直接使用 git 拉取了https://kernel ...

  2. redis的fd与epoll是怎么使用的

    Redis 的高性能网络模型核心依赖于 文件描述符(fd) 和 epoll 的协同工作.下面我将从底层机制到实际应用,详细解析它们的配合方式: 一.核心组件关系图 二.fd 在 Redis 中的具体应 ...

  3. kettle介绍-Step之If field value is null

    If field value is null介绍 替换NULL值步骤可以将输入流中所有字段的空值进行替换,也可以指定一种类型下的空值进行替换,还可以指定一个字段下的空值进行替换 三种替换NULL模式 ...

  4. 通过远程连接,docker访问获取数据表信息

    <dependency> <groupId>com.jcraft</groupId> <artifactId>jsch</artifactId&g ...

  5. 在使用import win32api时,报错:No module named win32api

    二.在使用import win32api时,报错:No module named win32api 网上查到有下面解决办法: pip install pypiwin32 或 pip3 install ...

  6. [随记]-linux侦听端口的4种方法

    侦听 192.168.0.1 服务器上的 10086 端口是否打开 1. telnet telnet是windows 内置的功能,当然 linux 也有.用法:  tenlet 192.168.0.1 ...

  7. 关于正点原子input子系统,驱动中按键中断只检测了上升或下降沿却可以实现连按(EV_REP)的原因

    问题 在学习到Linux内核input子系统时,产生了一个疑惑.可以看到,我们改造按键中断驱动程序(请见keyinputdriver.c(内核驱动代码)),通过检测按键的上升沿和下降沿,在中断处理函数 ...

  8. SQL 强化练习 (十一)

    sql 冲冲冲.... 也没啥可犹豫, 作为一名数据分析师, 必须掌握的技能, 就要熟练到写 Python 那样的感觉, 就应该可以了, 但目前还是差的比较远, 原因是, 没有相关的一些比较复杂一些的 ...

  9. Java Solon v3.3.0 发布(国产优秀应用开发基座)

    Solon 框架! Solon 是新一代,Java 企业级应用开发框架.从零开始构建(No Java-EE),有灵活的接口规范与开放生态.采用商用友好的 Apache 2.0 开源协议,是" ...

  10. TVMC python:一种TVM的高级API

    Step 0: Imports from tvm.driver import tvmc Step 1: Load a model 下载模型: wget https://github.com/onnx/ ...