StringBuilder、StringBuffer源码分析

StringBuilder源码分析

类结构

public final class StringBuilder
extends AbstractStringBuilder
implements java.io.Serializable, CharSequence

StringBuilder使用final关键字修饰,和String一样不可以被继承

StringBuilder继承AbstractStringBuilder并实现了Serializable和CharSequence,可以被序列化

方法

StringBuilder 的方法多是直接调用父类AbstractStringBuilder的方法,这里找几个典型的方法看一下

StringBuilder append(Object obj)方法重写父类的方法,追加Object类型的元素

@Override
public StringBuilder append(Object obj) {
return append(String.valueOf(obj));//String.valueOf(obj)获取对象转换成的字符串
}
public static String valueOf(Object obj) {
return (obj == null) ? "null" : obj.toString();
}
@Override
public StringBuilder append(String str) {
super.append(str);
return this;
}
public AbstractStringBuilder append(String str) {
if (str == null)
return appendNull();//如果为null追加字符串“null”
int len = str.length();
ensureCapacityInternal(count + len);
//拷贝字符串到数组
str.getChars(0, len, value, count);
count += len;
return this;
}

StringBuilder delete(int start, int end)删除指定起点下标到指定结束下标的字符

@Override
public StringBuilder delete(int start, int end) {
super.delete(start, end);
return this;
}
public AbstractStringBuilder delete(int start, int end) {
if (start < 0)
throw new StringIndexOutOfBoundsException(start);
if (end > count)//如果结束下标>当前保存char的最大下标,直接赋值为最大下标
end = count;
if (start > end)
throw new StringIndexOutOfBoundsException();
int len = end - start;
if (len > 0) {
//把删除尾下标后的元素拷贝到删除起始下标后
System.arraycopy(value, start+len, value, start, count-end);
count -= len;
}
return this;
}

StringBuilder replace(int start, int end, String str)使用字符串替换指定范围内的字符

@Override
public StringBuilder replace(int start, int end, String str) {
super.replace(start, end, str);
return this;
}
public AbstractStringBuilder replace(int start, int end, String str) {
if (start < 0)
throw new StringIndexOutOfBoundsException(start);
if (start > count)
throw new StringIndexOutOfBoundsException("start > length()");
if (start > end)
throw new StringIndexOutOfBoundsException("start > end"); if (end > count)
end = count;
int len = str.length();
//计算需要的容量
int newCount = count + len - (end - start);
//扩容
ensureCapacityInternal(newCount);
//删除指定范围的字符
System.arraycopy(value, end, value, start + len, count - end);
//在删除的起始位置插入字符串
str.getChars(value, start);
count = newCount;
return this;
}

StringBuilder insert(int offset, Object obj)在指定位置插入对象

@Override
public StringBuilder insert(int offset, Object obj) {
super.insert(offset, obj);
return this;
}
public AbstractStringBuilder insert(int offset, Object obj) {
return insert(offset, String.valueOf(obj));//String.valueOf(obj)获取对象转换的字符串
}
public static String valueOf(Object obj) {
return (obj == null) ? "null" : obj.toString();
}
@Override
public StringBuilder insert(int offset, String str) {
super.insert(offset, str);
return this;
}
public AbstractStringBuilder insert(int offset, String str) {
if ((offset < 0) || (offset > length()))
throw new StringIndexOutOfBoundsException(offset);
if (str == null)
str = "null";
int len = str.length();
//扩容
ensureCapacityInternal(count + len);
//把要插入位置后一定数量的字符(插入字符串长度)串移动后移一定距离(插入字符串长度)
System.arraycopy(value, offset, value, offset + len, count - offset);
//插入要插入的字符串
str.getChars(value, offset);
count += len;
return this;
}

可以看到,StringBuilder的append、insert、replace、delete都是对父类的char数组进行的一些操作,并没有产生新的对象

String toString() 最精髓的一个方法

@Override
public String toString() {
//把进过一些列修改后的最终char数组生成String
return new String(value, 0, count);
}

这里我们看到在toString的时候,把char数组生成了String,这也是为什么StringBuilder比String效率高的原因,String类没做一点修改都会生成新的对象,那么在频繁拼串和截取字符串时,效率当然不如StringBuilder

StringBuffer源码分析

类结构

public final class StringBuffer
extends AbstractStringBuilder
implements java.io.Serializable, CharSequence

StringBuffer的类结构和StringBuilder的一样

方法

StringBuffer和StringBuilder一样,很多方法都是调用父类AbstractStringBuilder的方法,我们看几个最主要的方法

StringBuffer append(Object obj)向StringBuffer中追加对象,和StringBuilder的追加对象一样的代码

@Override
public synchronized StringBuffer append(Object obj) {
toStringCache = null;
super.append(String.valueOf(obj));
return this;
}
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;
}
public void getChars(int srcBegin, int srcEnd, char dst[], int dstBegin) {
if (srcBegin < 0) {
throw new StringIndexOutOfBoundsException(srcBegin);
}
if (srcEnd > value.length) {
throw new StringIndexOutOfBoundsException(srcEnd);
}
if (srcBegin > srcEnd) {
throw new StringIndexOutOfBoundsException(srcEnd - srcBegin);
}
System.arraycopy(value, srcBegin, dst, dstBegin, srcEnd - srcBegin);
}

StringBuffer delete(int start, int end)删除指定范围内的字符,和StringBuilder中delete方法代码一样

@Override
public synchronized StringBuffer delete(int start, int end) {
toStringCache = null;
super.delete(start, end);
return this;
}
public AbstractStringBuilder delete(int start, int end) {
if (start < 0)
throw new StringIndexOutOfBoundsException(start);
if (end > count)
end = count;
if (start > end)
throw new StringIndexOutOfBoundsException();
int len = end - start;
if (len > 0) {
System.arraycopy(value, start+len, value, start, count-end);
count -= len;
}
return this;
}

StringBuffer replace(int start, int end, String str)方法使用字符串替换指定范围内的字符,和StringBuilder的replace方法代码一样

@Override
public synchronized StringBuffer replace(int start, int end, String str) {
toStringCache = null;
super.replace(start, end, str);
return this;
}
public AbstractStringBuilder replace(int start, int end, String str) {
if (start < 0)
throw new StringIndexOutOfBoundsException(start);
if (start > count)
throw new StringIndexOutOfBoundsException("start > length()");
if (start > end)
throw new StringIndexOutOfBoundsException("start > end"); if (end > count)
end = count;
int len = str.length();
int newCount = count + len - (end - start);
ensureCapacityInternal(newCount); System.arraycopy(value, end, value, start + len, count - end);
str.getChars(value, start);
count = newCount;
return this;
}

StringBuffer insert(int offset, Object obj)在指定位置插入字符串,也是和StringBuilder的insert方法代码一样

 @Override
public synchronized StringBuffer insert(int offset, Object obj) {
toStringCache = null;
super.insert(offset, String.valueOf(obj));
return this;
}
public AbstractStringBuilder insert(int offset, String str) {
if ((offset < 0) || (offset > length()))
throw new StringIndexOutOfBoundsException(offset);
if (str == null)
str = "null";
int len = str.length();
ensureCapacityInternal(count + len);
System.arraycopy(value, offset, value, offset + len, count - offset);
str.getChars(value, offset);
count += len;
return this;
}

通过分析这几个方法源码,我们可以看到,StringBuilder和StringBuffer在方法的实现上是一致的,唯一的区别是StringBuffer的所有方法都加了synchronized锁,所以是线程安全的

String toString()把StringBuffer转换成字符串

@Override
public synchronized String toString() {
if (toStringCache == null) {
toStringCache = Arrays.copyOfRange(value, 0, count);
}
return new String(toStringCache, true);
}

StringBuffer与StringBuilder都是在修改的时候并没有产生新的对象,只是在调用toString方法是才转换为字符串。

总结

  1. StringBuilder和StringBuffer的类结构是一致的,都是使用父类的char数组保存字符。
  2. StringBuffer的所有方法都加了synchronized锁,所以是线程安全的,但是这也使得它的效率比StringBuilder低。
  3. StringBuilder和StringBuffer的基本思想是一致的,对StringBuilder、StringBuffer的任何修改都不会产生新对象,这也使得StringBuilder、StringBuffer在进行大量拼串截取时比String的效率高。

StringBuilder、StringBuffer分析比较的更多相关文章

  1. [面试]StringBuilder StringBuffer源码粘贴(并非源码分析, 请绕道)

    StringBuilder StringBuffer 源码 这里只是我个人笔记, 并没有在这篇文章里做任何技术分享, 请同学们绕道- -.如果需要我会后期补上的..... 本来是想分析源码来着, 但是 ...

  2. JDK源码分析系列---String,StringBuilder,StringBuffer

    JDK源码分析系列---String,StringBuilder,StringBuffer 1.String public final class String implements java.io. ...

  3. stringbuffer与stringbuilder区别分析

    它们到底都有什么区别呢! 三者都是用来对字符串进行操作,String通常用来定义一个变量,而StringBuilder StringBuffer则通常用来对字符串进行拼接等操作.但其实String同样 ...

  4. string,stringbuilder,stringbuffer用法

    总结:1.如果要操作少量的数据用 = String   ==================================>字符串常量2.单线程操作字符串缓冲区 下操作大量数据 = Strin ...

  5. 深入源码剖析String,StringBuilder,StringBuffer

    [String,StringBuffer,StringBulider] 深入源码剖析String,StringBuilder,StringBuffer [作者:高瑞林] [博客地址]http://ww ...

  6. String&StringBuilder&StringBuffer总结

    一.String的JVM内存分配测试与分析           String a="a"; String b="b"; String c="ab&qu ...

  7. 数据结构(逻辑结构,物理结构,特点) C#多线程编程的同步也线程安全 C#多线程编程笔记 String 与 StringBuilder (StringBuffer) 数据结构与算法-初体验(极客专栏)

    数据结构(逻辑结构,物理结构,特点) 一.数据的逻辑结构:指反映数据元素之间的逻辑关系的数据结构,其中的逻辑关系是指数据元素之间的前后件关系,而与他们在计算机中的存储位置无关.逻辑结构包括: 集合 数 ...

  8. [java] StringBuilder / StringBuffer / String 建立字符串

    1.三者在建立新字符串速度方面的比较: StringBuilder >  StringBuffer  >  String 2.线程安全性: StringBuilder:线程非安全的 Str ...

  9. JAVA StringBuilder StringBuffer String比较

    字符串连接时的效率: StringBuilder>StringBuffer>String 区别: StringBuilder效率最高,但是不是线程安全的,适用于单线程.多线程用String ...

随机推荐

  1. CodeForces 506B/505D Mr. Kitayuta's Technology

    Portal:http://codeforces.com/problemset/problem/506/B http://codeforces.com/problemset/problem/505/D ...

  2. 聊起 BigTable,让你不再胆怯

    谷歌“三驾马车”的出现,才真正把我们带入了大数据时代,并指明了大数据的发展方向. GFS 作为其中一驾宝车,解决了大数据存储的难题.它能够把大量廉价的普通机器,聚在一起,充分让每台廉价的机器发挥光和热 ...

  3. javascript 3d网页 示例 ( three.js 初探 七)

    1 完整代码下载 https://pan.baidu.com/s/1JJyVcP2KqXsd5G6eaYpgHQ 提取码 3fzt (压缩包名: 2020-4-5-demo.zip) 2 图片展示 3 ...

  4. Base64编码与解码原理

    Base64编码是使用64个可打印ASCII字符(A-Z.a-z.0-9.+./)将任意字节序列数据编码成ASCII字符串,另有“=”符号用作后缀用途. base64索引表 base64编码与解码的基 ...

  5. 《Three.js 入门指南》3.1.1 - 基本几何形状 - 平面(PlaneGeometry)

    3.1 基本几何形状 平面(PlaneGeometry) 说明: 这里的平面(PlaneGeometry)其实是一个长方形,而不是数学意义上无限大小的平面. 构造函数: THREE.PlaneGeom ...

  6. Linux命令后面加 & 的作用

    在命令的后面加一个 & 的作用是,将这个任务放到后台执行.看下面的例子. 输入gedit回车,可以看到,打开了Linux的文本编辑器,但是命令窗口执行不了其他命令了,只有退出文本编辑器才能继续 ...

  7. Oracle 12C安装教程

    Oracle 12C安装教程 由 Alma 创建, 最后一次修改 2018-06-04 10:32:31 Oracle 12c,全称Oracle Database 12c,是Oracle 11g的升级 ...

  8. 虚拟机的vmnet8网卡找不到了

    不知道我设置了什么,在我于Linux中配置网络时发现怎么都不行,检查了一下发现用于NAT的网卡没有了. 我重启了电脑之后发现还是没有. 于是按照网上的办法在虚拟网络编辑器将其重置,如下图. 问题解决. ...

  9. MTK Android ListPreference的用法

    首先,我们明确,preference是和数据存储相关的. 其次,它能帮助我们方便的进行数据存储!为什么这个地方一定要强调下方便的这个词呢?原因是,我们可以根本就不使用,我们有另外的N种办法可以实现同样 ...

  10. MySQL入门,第五部分,表结构的修改

    ALTER TABLE <基本表名> [ ADD <新列名> <列数据类型> [列完整性约束] DROP COLUMN <列名> MODIFY < ...