JDK源码解析之 Java.lang.StringBuilder
StringBuilder类表示一个可变的字符序列。StringBuilder的API与StringBuffer互相兼容,但是StringBuilder是非线程安全的,在大多数实现中它比StringBuffer更快。
一、类定义
public final class StringBufferextends AbstractStringBuilderimplements java.io.Serializable, CharSequence
{...
}
StringBuilder类被 final 所修饰,因此不能被继承。
StringBuilder类继承于 AbstractStringBuilder类。实际上,AbstractStringBuilder类具体实现了可变字符序列的一系列操作,比如:append()、insert()、delete()、replace()、charAt()方法等。值得一提的是,StringBuffer也是继承于AbstractStringBuilder类。
StringBuilder类实现了2个接口:
- Serializable 序列化接口,表示对象可以被序列化。
- CharSequence 字符序列接口,提供了几个对字符序列进行只读访问的方法,比如:length()、charAt()、subSequence()、toString()方法等。
二、成员变量
private transient char[] toStringCache;// AbstractStringBuilder.javachar[] value;int count;
- value、count这两个变量是继承自父类
三、构造方法
//默认构造方法设置了value数组的初始容量为16。
public StringBuilder() {super(16);
}
//设置了value数组的初始容量为指定的大小。
public StringBuilder(int capacity) {super(capacity);
}
//接受一个String对象作为参数,设置了value数组的初始容量为String对象的长度+16,并把String对象中的字符添加到value数组中。
public StringBuilder(String str) {super(str.length() + 16);append(str);
}
//接受一个CharSequence对象作为参数,设置了value数组的初始容量为CharSequence对象的长度+16,并把CharSequence对象中的字符添加到value数组中。
public StringBuilder(CharSequence seq) {this(seq.length() + 16);append(seq);
}// AbstractStringBuilder.java
AbstractStringBuilder(int capacity) {value = new char[capacity];
}
StringBuilder类提供了4个构造方法。构造方法主要完成了对value数组的初始化。
四、普通方法
StringBuilder实现了AbstractStringBuilder和CharSequence,他的方法都来自于这两个类,绝大部分都是通过super来调用的。
4.1、append()方法
@Override
public StringBuilder append(boolean b) {super.append(b);return this;
}// AbstractStringBuilder.java
public AbstractStringBuilder append(boolean b) {if (b) {ensureCapacityInternal(count + 4);value[count++] = 't';value[count++] = 'r';value[count++] = 'u';value[count++] = 'e';} else {ensureCapacityInternal(count + 5);value[count++] = 'f';value[count++] = 'a';value[count++] = 'l';value[count++] = 's';value[count++] = 'e';}return this;
} @Override
public StringBuilder append(String str) {super.append(str);return this;
}// AbstractStringBuilder.java
public AbstractStringBuilder append(String str) {if (str == null)return appendNull();int len = str.length();ensureCapacityInternal(count + len);str.getChars(0, len, value, count);count += len;return this;
}
调用了父类AbstractStringBuilder类中对应的方法。最后,append()方法返回了StringBuilder对象自身。
append()方法将指定参数类型的字符串表示形式追加到字符序列的末尾。它可以接受boolean、char、char[]、CharSequence、double、float、int、long、Object、String、StringBuffer这些类型的参数。这些方法最终都,以便用户可以链式调用StringBuilder类中的方法。
AbstractStringBuilder类的各个append()方法大同小异。append()方法在追加字符到value数组中之前都会调用ensureCapacityInternal()方法来确保value数组有足够的容量,然后才把字符追加到value数组中。
4.2、toString()方法
@Override
public String toString() {// Create a copy, don't share the arrayreturn new String(value, 0, count);
}
4.3、writeObject和readObject
/*** Save the state of the {@code StringBuilder} instance to a stream* (that is, serialize it).** @serialData the number of characters currently stored in the string* builder ({@code int}), followed by the characters in the* string builder ({@code char[]}). The length of the* {@code char} array may be greater than the number of* characters currently stored in the string builder, in which* case extra characters are ignored.*/private void writeObject(java.io.ObjectOutputStream s)throws java.io.IOException {s.defaultWriteObject();s.writeInt(count);s.writeObject(value);}/*** readObject is called to restore the state of the StringBuffer from* a stream.*/private void readObject(java.io.ObjectInputStream s)throws java.io.IOException, ClassNotFoundException {s.defaultReadObject();count = s.readInt();value = (char[]) s.readObject();}
实现自Serializable接口后,可定制的序列化过程
五、StringBuffer 与 StringBuilder
5.1、类继承关系:
StringBuilder 和 StringBuffer是高度类似的两个类,都是可变字符序列,他们都实现了AbstractStringBuilder
5.2、方法体系:
他们除了实现了AbstractStringBuilder 和 CharSequence外,实际上他们没有自己的方法,所有的方法都来自AbstractStringBuilder 和 CharSequence
5.3、线程安全:
StringBuffer是线程安全的,StringBuilder是非线程安全的,其实线程安全也就是方法前面增加了一个synchronized关键字
toStringCache:StringBuffer中有一个toStringCache 就像它的名字一样,toString()方法的cache
简言之就是缓存toString方法的,每次调用toString会检查这个字段,如果不为null将会使用它进行对象创建
如果为null 将会给他初始化赋值,也就是缓存,当调用其他的任何方法改变StringBuffer时,就会把toStringCache进行清空,如果每次都是更改变动后调用,显然,还适得其反的浪费了性能
如果多次调用toString将会得到好处
5.4、总结
两个类的功能逻辑上来说基本一样,都是可变的字符序列
代码的相似度也很高
他们本身就是为了做同一件事情
只不过是各自的侧重点不同
他们都实现了AbstractStringBuilder和CharSequence
他们的方法都来自于这两个类
只不过StringBuffer是线程安全的,StringBuilder非线程安全
其实 早在1.0版本StringBuffer 就已经存在了
StringBuffer则是在1.5才加入进来的,AbstractStringBuilder 也是在1.5加入进来
StringBuilder 就是 StringBuffer的一个非线程安全的实现
AbstractStringBuilder 也是后来才对类的设计进行抽象升华的
StringBuffer才实现了这个类
他们的源代码也大多数是雷同的
主要差异就在于以下三点
StringBuffer覆盖的方法略微多一点
StringBuffer 在方法上增加了synchronized关键字用于同步,亦或者应该说,StringBuilder去掉了synchronized
StringBuffer的toStringCache缓存
如果去掉这三点,这两份代码就几乎是一样的了
除非你的确非常确信你需要使用StringBuffer
否则,如果不可变使用String
如果可变使用StringBuilder ,尽可能的放弃StringBuffer 吧
总结起来就一句话
StringBuilder是StringBuffer的非同步版本就是版本改写
能用StringBuilder就不要用StringBuffer。
JDK源码解析之 Java.lang.StringBuilder相关推荐
- JDK源码解析之 java.lang.Exception
异常.是所有异常的基类,用于标识一般的程序运行问题.这些问题通常描述一些会被应用程序捕获的反常情况. 一.源码部分 //继承了java.lang.Throwable public class Exce ...
- JDK源码解析之 java.lang.Error
java.lang.Error 错误.是所有错误的基类,用于标识严重的程序运行问题.这些问题通常描述一些不应被应用程序捕获的反常情况. 一.源码部分 //继承了java.lang.Throwable ...
- JDK源码解析之 java.lang.Thread
位于java.lang包下的Thread类是非常重要的线程类,它实现了Runnable接口,今天我们来学习一下Thread类,在学习Thread类之前,先介绍与线程相关知识:线程的几种状态.上下文切换 ...
- JDK源码解析之 java.lang.Integer
teger 基本数据类型int 的包装类 Integer 类型的对象包含一个 int 类型的字段 一.类定义 public final class Integer extends Number imp ...
- JDK源码解析之 Java.lang.String
String 类代表字符串.Java 程序中的所有字符串字面值(如 "abc" )都作为此类的实例实现. 字符串是常量:它们的值在创建之后不能更改.字符串缓冲区支持可变的字符串.因 ...
- JDK源码解析之 Java.lang.Object
Object类是Java中其他所有类的祖先,没有Object类Java面向对象无从谈起.作为其他所有类的基类,Object具有哪些属性和行为,是Java语言设计背后的思维体现. Object类位于ja ...
- JDK源码解析之 Java.lang.Compiler
Compiler类提供支持Java到本机代码编译器和相关服务.在设计上,它作为一个占位符在JIT编译器实现. 一.源码部分 public final class Compiler {private C ...
- JDK源码解析之 Java.lang.Byte
byte,即字节,由8位的二进制组成.在Java中,byte类型的数据是8位带符号的二进制数,以二进制补码表示的整数 取值范围:默认值为0,最小值为-128(-27);最大值是127(27-1) By ...
- JDK源码解析之 Java.lang.AbstractStringBuilder
这个抽象类是StringBuilder和StringBuffer的直接父类,而且定义了很多方法,因此在学习这两个类之间建议先学习 AbstractStringBuilder抽象类 该类在源码中注释是以 ...
最新文章
- 一行代码都不写!Github 博客小白版入门教程
- python使用imbalanced-learn的OneSidedSelection方法进行下采样处理数据不平衡问题
- IIS部署ASP.Net Core 502.5错误和解决
- 花5分钟过一遍jar包和war包的区别,以后都不会再迷茫
- python条件表达式三门课至少有一门及格_Python/ MySQL练习题(一)
- 开发软件不是闭卷考试
- linux远程拷贝免手动输入密码,scp远程拷贝避免输入密码
- vue服务端渲染添加缓存
- (软件工程复习核心重点)第十章面向对象设计-第四节:设计人机交互子系统和设计任务管理子系统
- 《System语言详解》——1.SystemTap概述
- 文件已被占用的解决方法
- 基于深度学习的视频质量分析 深度学习视频质量诊断 图像质量诊断
- 为什么程序员做外包会被瞧不起
- PVT(Pyramid Vision Transformer: A Versatile Backbone for Dense Prediction without Convolutions)
- Android 安卓开发板 读取串口
- 数学建模比赛经验总结
- XTransfer技术专家康康:从普通程序员到架构师的进化之路
- Oracle11g for Windows
- 第三方组件提权-SQL server提权
- 【对讲机的那点事】带你玩转宝锋UV6R对讲机(三)
热门文章
- 求二叉树中结点个数代码java_求二叉树中第K层结点的个数
- Qt中多个动态创建的按钮同时绑定一个槽函数,判断被点击的是哪个按钮
- matlab保存并关闭excel文件夹,[转载]Matlab批量操作目标文件夹下的Excel文件
- 脉冲宽度调制pdm_PWM (脉冲宽度调制)原理与实现
- python boxplot orient_Python 可视化 | Seaborn5 分钟入门 (三)——boxplot 和 violinplot
- 控制cpu_I/O控制器及控制方式,了解一下
- 苹果关掉200m限制_苹果下载大于200M限制
- oracle00333,Oracle数据库REDO损坏ora-00333修复手札
- android音乐播放器案例,Android MediaPlayer实现音乐播放器实例代码
- c语言编程常见问题解答,C语言编程常见问题解答之常用函数的包含文件