我的jdk源码(五):StringBuilder 高效可多次修改String
一、概述
StringBuilder类和StringBuffer类一样,也是我们动态操作字符串常用到的类,jdk1.8中StringBuilder同样继承了父类AbstractStringBuilder类,并且在源码内很多方法都是直接调用的父类AbstractStringBuilder的方法。和 StringBuffer类不同的是,StringBuilder 类的没有了字符缓存数组,成员方法也没有被synchronized关键字修饰,所以StringBuilder是线程不安全的,但是效率会更高,这也是与StringBuffer类最大的区别。接下来就让我们进入到StringBuffer的源码学习!
二、源码分析
(1) 类的声明,源码如下:
public final class StringBuilder
extends AbstractStringBuilder
implements java.io.Serializable, CharSequence
如源码所示,StringBuilder类也是被final关键字修饰,为最终类,无法被继承 。StringBuilder类不但继承了AbstractStringBuilder类,还实现了Serializable接口和CharSequence接口。实现Serializable是为了能序列化它的对象,具体在《》一文中有详细介绍;实现CharSequence是为了代表该类,或其子类是一个字符序列,具体在《》一文中有详细介绍 。
(2) 成员变量
static final long serialVersionUID = 4383685877147921099L;
与StringBuffer最大的区别就是成员变量由于不是线程安全的,所以也不需要设置字符缓存,所以没有字符数组toStringCache。
(3) 构造函数
public StringBuilder() {
super(16);
}
public StringBuilder(int capacity) {
super(capacity);
}
public StringBuilder(String str) {
super(str.length() + 16);
append(str);
}
public StringBuilder(CharSequence seq) {
this(seq.length() + 16);
append(seq);
}
构造函数与StringBuffer几乎一致,都是调用父类构造方法,并且很直观的反映了以下几点:
* 无参默认创建容量为16的底层数组。
* 传入字符串时,容量为字符串长度+16。即StringBuilder sb = new StringBuilder("abc");时,sb.length()值为3,sb.capacity()值为19。
* 传入整数时,则设定容量为参数值 。
(4) toString()方法源码如下:
public String toString() {
//直接用字符数组new一个新的String对象
return new String(value, 0, count);
}
(5) writeObject()方法源码如下:
private void writeObject(java.io.ObjectOutputStream s)
throws java.io.IOException {
s.defaultWriteObject();
s.writeInt(count);
s.writeObject(value);
}
在进行序列化的时候保存StringBuilder对象的状态到一个流中。
(6) readObject()方法源码如下:
private void readObject(java.io.ObjectInputStream s)
throws java.io.IOException, ClassNotFoundException {
s.defaultReadObject();
count = s.readInt();
value = (char[]) s.readObject();
}
反序列化时从流中获取StringBuild对象序列化之前的状态。
三、总结
StringBuilder类与StringBuffer类大致相同,记住最大的不同就是StringBuffer类成员方法采用了同步关键字synchronized修饰,所以是线程安全的,但是也导致效率要低一些。那么关于String家族的类就介绍到这里,接下来进入线程相关的类学习中,敬请期待《我的jdk源码(六):Thread 》。