Java开发笔记(一百五十四)StringBuffer和StringBuilder的由来
本来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的由来的更多相关文章
- Java开发笔记(五十四)内部类和嵌套类
通常情况下,一个Java代码文件只定义一个类,即使两个类是父类与子类的关系,也要把它们拆成两个代码文件分别定义.可是有些事物相互之间密切联系,又不同于父子类的继承关系,比如一棵树会开很多花朵,这些花儿 ...
- Java开发笔记(三十四)字符串的赋值及类型转换
不管是基本的char字符型,还是包装字符类型Character,它们的每个变量只能存放一个字符,无法满足对一串字符的加工.为了能够直接操作一连串的字符,Java设计了专门的字符串类型String,该类 ...
- Java开发笔记(五十)几种开放性修饰符
前面介绍子类继承父类的时候,提到了public(公共)和private(私有)两个修饰符,其中public表示它所修饰的实体是允许外部访问的:而private表示它所修饰的实体不允许外部访问,只能在当 ...
- Java开发笔记(五十六)利用枚举类型实现高级常量
前面介绍了联合利用final和static可实现常量的定义,该方式用于简单的常量倒还凑合,要是用于复杂的.安全性高的常量,那就力不从心了.例如以下几种情况,final结合static的方式便缺乏应对之 ...
- Java开发笔记(八十四)文件与目录的管理
程序除了处理内存中的数据结构,还要操作磁盘上的各类文件,这里的磁盘是个统称,泛指可以持久保留数据的存储介质,包括但不限于:插在软驱中的软盘.固定在机箱中的硬盘.插在光驱中的光盘.插在USB接口上的U盘 ...
- Java开发笔记(二十四)方法的组成形式
经过前面的学习,我们发现演示的Java代码越来越复杂,而且每个例子的代码都堆在入口方法main内部,这会导致如下问题:1.一个方法内部堆砌了太多的代码行,看着费神,维护起来也吃力:2.部分代码描述的是 ...
- Java开发笔记(五十二)对象的类型检查
前面介绍了类的多态性,来自于鸡类的实例chicken,既能用来表达公鸡实例,也能用来表达母鸡实例.可是这导致了一个问题,假如在call方法内部需要手工判断输入参数属于公鸡实例还是母鸡实例,那该如何是好 ...
- Java开发笔记(五十五)关键字static的用法
前面介绍嵌套类的时候讲到了关键字static,用static修饰类,该类就变成了嵌套类.从嵌套类的用法可知,其它地方访问嵌套类之时,无需动态创建外层类的实例,直接创建嵌套类的实例就行.其实static ...
- Java开发笔记(五十八)简单接口及其实现
前面介绍了抽象方法及抽象类的用法,看似解决了不确定行为的方法定义,既然叫唤动作允许声明为抽象方法,那么飞翔.游泳也能声明为抽象方法,并且鸡类涵盖的物种不够多,最好把这些行为动作扩展到鸟类这个群体,于是 ...
- Java开发笔记(五十九)Java8之后的扩展接口
前面介绍了接口的基本用法,有心的朋友可能注意到这么一句话“在Java8以前,接口内部的所有方法都必须是抽象方法”,如此说来,在Java8之后,接口的内部方法也可能不是抽象方法了吗?之所以Java8对接 ...
随机推荐
- .NET中的线程本地存储(TLS)与AsyncLocal(一)
一.TLS 线程本地存储(Thread Local Storage),字面意思就是专属某个线程的存储空间.变量大体上分为全局变量和局部变量,一个进程中的所有线程共享地址空间,这个地址空间被划分为几个固 ...
- 🎀文件对比工具--BeyondCompare
简介 Beyond Compare 是一款功能强大的文件和文件夹比较工具,由Scooter Software开发.它可以帮助用户轻松地比较文件和文件夹的差异,并且可以合并变化.同步文件以及备份重要数据 ...
- python调用百度ocr接口,实现图片内文字识别
第一步,到百度智能云申请接口资源 打开地址:https://cloud.baidu.com/?from=console,点击产品下的通用场景文字识别 立即使用,跳转页领取免费资源(土豪可直接购买) 选 ...
- 说说 Java 的执行流程?
Java 的执行流程 Java 的执行流程包括多个阶段,从源码编写到最终程序的执行,涉及到编译.类加载.字节码执行.垃圾回收等多个环节.下面将详细介绍 Java 程序的执行流程. 1. 编写源代码 开 ...
- EF Core 中避免 SQL 注入的三种写法
SQL 注入攻击可能会对我们的应用程序产生严重影响,导致敏感数据泄露.未经授权的访问和应用程序受损.EF Core 提供了三种内置机制来防止 SQL 注入攻击. 1.利用 LINQ 查询语法和参数化查 ...
- GUI development with Rust and GTK4 阅读笔记
简记 这是我第二次从头开始阅读,有第一次的印象要容易不少. 如果只关心具体的做法,而不思考为什么这样做,以及整体的框架,阅读的过程将会举步维艰. 简略记录 gtk-rs 的书中提到的点.对同一个问题书 ...
- 中国版 Cursor:CodeBuddy
我正在参加CodeBuddy「首席试玩官」内容创作大赛,本文所使用的 CodeBuddy 免费下载链接:腾讯云代码助手 CodeBuddy - AI 时代的智能编程伙伴 一句话即可让 AI 助手生成网 ...
- WPF在Visual studio中打包,发布注意事项
右键项目,发布的时候,需要选择独立,否则依赖库的话,有可能客户端没有.net core或.net framework,那么就会报错,提醒对方下载. 今天发现,VS 2022,直接Debug运行后,Re ...
- deepseek-r1的1.5b、7b、8b、14b、32b、70b和671b有啥区别?
DeepSeek-R1系列提供了多种参数规模的模型(1.5B.7B.8B.14B.32B.70B 和 671B),它们在模型架构.性能表现.资源需求和适用场景上有显著差异.以下是对这些版本的核心区别总 ...
- frp实现内网穿透访问内网多台Linux服务器
本文主要记录笔者在使用frp实现内网穿透访问内网多台Linux服务器的全过程,包括公网服务器的配置.frp服务端.客户端的下载与配置,以及配置systmctl来实现系统级启停frp,并记录我遇到的一些 ...