JDK源码解析之 Java.lang.AbstractStringBuilder

这个抽象类是StringBuilder和StringBuffer的直接父类,而且定义了很多方法,因此在学习这两个类之间建议先学习 AbstractStringBuilder抽象类

该类在源码中注释是以JDK1.5开始作为前两个类的父类存在的,可是直到JDK1.8的API中,关于StringBuilder和StringBuffer的父类还是Object

一、类定义

abstract class AbstractStringBuilder implements Appendable, CharSequence {}

类名用abstract修饰说明是一个抽象类,只能被继承,不能直接创建对象。但是它就一个抽象方法,toString方法。

实现了两个接口:

  • CharSequence:这个字符序列的接口已经很熟悉了,用来表示一个有序字符的集合

  • ApAppendable接口能够被追加 char 序列和值的对象。如果某个类的实例打算接收来自 Formatter 的格式化输出,那么该类必须实现 Appendable 接口。

二、成员变量

/*** The value is used for character storage.*/
char[] value;/*** The count is the number of characters used.*/
int count;

和String中的变量不同,内部的char[] value不再是final的了,也就意味着可变

三、构造方法

    /** * This no-arg constructor is necessary for serialization of subclasses.*/AbstractStringBuilder() {}/** * Creates an AbstractStringBuilder of the specified capacity.*/AbstractStringBuilder(int capacity) {value = new char[capacity];}

AbstractStringBuilder提供两个构造方法,一个是无参构造方法。一个是传一个capacity(代表数组容量)的构造,这个构造方法用于指定类中value数组的初始大小,数组大小后面还可动态改变。

四、普通方法

AbstractStringBuilder中的方法可大致分为五类:对属性的控制,对Value中char值的增删改查。

4.1、属性控制

主要是对Value的长度与容量进行的操作

1.length():

返回已经存储的实际长度(就是count值)

public int length() {return count;}
2.capacity():

capacity这个单词是’容量’的意思,返回当前value可以存储的字符容量,即在下一次重新申请内存之前能存储字符序列的长度。

 public int capacity() {return value.length;}
3.ensureCapacity(int minimumCapacity):

确保value数组的容量是否够用,如果不够用,调用4.expandCapacity(minimumCapacity)方法扩容,参数为需要的容量

    public void ensureCapacity(int minimumCapacity) {if (minimumCapacity > 0)ensureCapacityInternal(minimumCapacity);}    
4.expandCapacity(int minimumCapacity):

对数组进行扩容,参数为需要的容量

 void expandCapacity(int minimumCapacity) {int newCapacity = (value.length + 1) * 2;if (newCapacity < 0) {newCapacity = Integer.MAX_VALUE;} else if (minimumCapacity > newCapacity) {newCapacity = minimumCapacity;}value = Arrays.copyOf(value, newCapacity);}
扩容的算法:

如果调用了该函数,说明容量不够用了,先将当前容量+1的二倍(newCapacity)与需要的容量(minimumCapacity)比较。
如果比需要的容量大,那就将容量扩大到容量+1的二倍;如果比需要的容量小,那就直接扩大到需要的容量。
使用Arrays.copyOf()这个非常熟悉的方法来使数组容量动态扩大

5.trimToSize():

如果value数组的容量有多余的,那么就把多余的全部都释放掉

public void trimToSize() {if (count < value.length) {value = Arrays.copyOf(value, count);}}
6.setLength(int newLength):

强制增大实际长度count的大小,如果 newLength 参数小于当前长度则长度将更改为指定的长度, 截断,数据不变;如果 newLength 参数大于或等于当前长度则将追加有效的 null 字符 (’\u0000’),使长度满足 newLength 参数

 public void setLength(int newLength) {if (newLength < 0)throw new StringIndexOutOfBoundsException(newLength);if (newLength > value.length)expandCapacity(newLength);if (count < newLength) {for (; count < newLength; count++)value[count] = '\0';
} else {count = newLength;}
}

4.2、获取方法

1. 代码点相关的五个方法:charAt(int) / codePointAt(int) / codePointBefore(int) / codePointCount(int, int) / offsetByCodePoints(int, int)

他们与String中的是一模一样的,代码也是一样的(就有个变量名变动)

2. getChars(int srcBegin, int srcEnd, char dst[],int dstBegin):

复制、将value[]的[srcBegin, srcEnd)拷贝到dst[]数组的desBegin开始处

public void getChars(int srcBegin, int srcEnd, char dst[],int dstBegin){if (srcBegin < 0)throw new StringIndexOutOfBoundsException(srcBegin);if ((srcEnd < 0) || (srcEnd > count))throw new StringIndexOutOfBoundsException(srcEnd);if (srcBegin > srcEnd)throw new StringIndexOutOfBoundsException("srcBegin > srcEnd");System.arraycopy(value, srcBegin, dst, dstBegin, srcEnd - srcBegin);}
3. 索引下标
 public int indexOf(String str) {return indexOf(str, 0);}public int indexOf(String str, int fromIndex) {return String.indexOf(value, 0, count, str, fromIndex);}

int indexOf(String str)、int indexOf(String str, int fromIndex)

第一次出现的指定子字符串在该字符串中的索引,可以指定索引

int lastIndexOf(String str)、int lastIndexOf(String str, int fromIndex)

返回最右边出现的指定子字符串在此字符串中的索引 ,也就是最后一个,可以指定索引,指定索引就从索引处 反向匹配

4. substring(int start, int end)

根据索引返回子串

    public String substring(int start, int end) {if (start < 0)throw new StringIndexOutOfBoundsException(start);if (end > count)throw new StringIndexOutOfBoundsException(end);if (start > end)throw new StringIndexOutOfBoundsException(end - start);return new String(value, start, end - start);}
5.String substring(int start)

substring(int start, int end)的简化方法,指定开始位置,默认结束位置为最后

6. CharSequence subSequence(int start, int end)

为了实现CharSequence方法,内部调用的substring

@Overridepublic CharSequence subSequence(int start, int end) {return substring(start, end);}

4.3、更新方法

更新方法比较少,因为是数组,数组的访问按照下标进行设置就好了,还提供了替换的功能,也算是更新操作

  1. setCharAt(int index, char ch)
public void setCharAt(int index, char ch) {if ((index < 0) || (index >= count))throw new StringIndexOutOfBoundsException(index);value[index] = ch;
}
2.AbstractStringBuilder replace(int start, int end, String str)

使用str替换对象中从start 开始到end结束的这一段

4.4、删除方法

1.AbstractStringBuilder delete(int start, int end)

删除指定范围的char

    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;}
2.AbstractStringBuilder deleteCharAt(int index)

删除某个位置的char

    public AbstractStringBuilder deleteCharAt(int index) {if ((index < 0) || (index >= count))throw new StringIndexOutOfBoundsException(index);System.arraycopy(value, index+1, value, index, count-index-1);count--;return this;}

4.5、添加方法

添加元素,分为尾部追加元素和中间插入元素,由于append与insert都为一系列方法,下列系列中的一部分方法

1.append(Object obj)

利用Object(或任何对象)的toString方法转成字符串然后添加到该value[]中

 public AbstractStringBuilder append(Object obj) {return append(String.valueOf(obj));}
2.append()的核心代码:append(String str)/append(StringBuffer sb)/append(CharSequence s)

直接修改value[],并且’添加’的意思为链接到原value[]的实际count的后面

 public AbstractStringBuilder append(String str) {if (str == null) str = "null";int len = str.length();if (len == 0) return this;int newCount = count + len;if (newCount > value.length)expandCapacity(newCount);str.getChars(0, len, value, count);count = newCount;return this;}// Documentation in subclasses because of synchro differencepublic AbstractStringBuilder append(StringBuffer sb) {if (sb == null)return append("null");int len = sb.length();int newCount = count + len;if (newCount > value.length)expandCapacity(newCount);sb.getChars(0, len, value, count);count = newCount;return this;}// Documentation in subclasses because of synchro differencepublic AbstractStringBuilder append(CharSequence s) {if (s == null)s = "null";if (s instanceof String)return this.append((String)s);if (s instanceof StringBuffer)return this.append((StringBuffer)s);return this.append(s, 0, s.length());}
3.insert(int index, char str[], int offset,int len):(insert的核心代码)

在value[]的下标为index位置插入数组str的一部分,该部分的范围为:[offset,offset+len);

 public AbstractStringBuilder insert(int index, char str[], int offset,int len){if ((index < 0) || (index > length()))throw new StringIndexOutOfBoundsException(index);if ((offset < 0) || (len < 0) || (offset > str.length - len))throw new StringIndexOutOfBoundsException("offset " + offset + ", len " + len + ", str.length " + str.length);int newCount = count + len;if (newCount > value.length)expandCapacity(newCount);System.arraycopy(value, index, value, index + len, count - index);System.arraycopy(str, offset, value, index, len);count = newCount;return this;}

4.6、其他方法

1. reverse()

按照字符进行翻转

public AbstractStringBuilder reverse() {boolean hasSurrogates = false;int n = count - 1;for (int j = (n-1) >> 1; j >= 0; j--) {int k = n - j;char cj = value[j];char ck = value[k];value[j] = ck;value[k] = cj;if (Character.isSurrogate(cj) ||Character.isSurrogate(ck)) {hasSurrogates = true;}}if (hasSurrogates) {reverseAllValidSurrogatePairs();}return this;}

五、总结

AbstractStringBuilder就是 可变 字符序列的一个纲领,

它规定了可变字符序列应该有的行为,

比如 添加字符/删除字符/更新字符/获取字符,

因为可变,所以对于可变的支持,自然是必不可少的,

另外,他作为String在很多方面的一个替代,必然也是提供了String的一些功能方法,

否则与String API 变化巨大 也是毫无意义,

因为毕竟本身就是为了描述字符序列。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/535790.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

RHEL下安装配置基于2台服务器的MYSQL集群

一、介绍这篇文档旨在介绍如何在RHEL下安装配置基于2台服务器的MySQL集群。并且实现任意一台服务器出现问题或宕机时MySQL依然能够继续运行。 注意&#xff01;虽然这是基于2台服务器的MySQL集群&#xff0c;但也必须有额外的第三台服务器作为管理节点&#xff0c;但这台服务器…

JDK源码解析之 Java.lang.StringBuffer

StringBuffer类表示一个可变的字符序列。StringBuffer的API与StringBuilder互相兼容&#xff0c;但是StringBuffer是线程安全的。在可能的情况下&#xff0c;建议优先使用StringBuilder&#xff0c;因为在大多数实现中它比StringBuffer更快。 一、类定义 public final class S…

redo和undo

这是在网上看到的对redo和undo的探讨&#xff1a; 1. redo 记录所有做过的事情&#xff0c;用于恢复 undo 记录事务的前镜相&#xff0c;用于回滚2. redo&#xff0c;恢复数据库时&#xff0c;按照重做日志文件来恢复你之前的操作 undo&#xff0c;撤消你做过的操作&#xff0…

JDK源码解析之 Java.lang.StringBuilder

StringBuilder类表示一个可变的字符序列。StringBuilder的API与StringBuffer互相兼容&#xff0c;但是StringBuilder是非线程安全的&#xff0c;在大多数实现中它比StringBuffer更快。 一、类定义 public final class StringBufferextends AbstractStringBuilderimplements ja…

从映射观点看索引

信息检索主要有“检”与“索&#xff08;办手续&#xff09;”两个动作。在图书馆借书时&#xff0c;一般而言&#xff0c; 找书的时间比办理手续的时间长得多&#xff0c;因而缩短检查时间是提高效率的关键。数据库中检 索信息也与此类似。 在没有索引文件时&#xff0c;DBM…

JDK源码解析之 Java.lang.Boolean

Boolean 类是将 boolean 基本类型进行包装。类型为 Boolean 的对象包含一个单一属性 value&#xff0c;其类型为 boolean。 此外还提供了许多将 boolean 转换为 String、String 转换为 boolean&#xff0c;以及其他一些方法。 一、类定义 public final class Boolean implemen…

MYSQL的集群的安装与配置(mysql-5.1.21)

具体安装与配置&#xff1a;1&#xff09;准备工作&#xff1a;6台机器&#xff0c;IP地址分别为192.168.0.&#xff08;231-236&#xff09;MGM节点&#xff1a;192.168.0.231(232)SQL 节点&#xff1a;192.168.0.233-234NDBD 节点&#xff1a;192.168.0.235-236系统都是REDHA…

JDK源码解析之 Java.lang.Byte

byte&#xff0c;即字节&#xff0c;由8位的二进制组成。在Java中&#xff0c;byte类型的数据是8位带符号的二进制数,以二进制补码表示的整数 取值范围&#xff1a;默认值为0&#xff0c;最小值为-128&#xff08;-27&#xff09;;最大值是127&#xff08;27-1&#xff09; Byt…

在命令行模式下管理SELinux

作者&#xff1a; Oslad.com (原创&#xff01;转载请注明出处) 2006-07-14 在 GUI 图形界面模式下&#xff0c;要更改 SELinux 的策略使用方式&#xff0c;只需依次点击“应用程序”&#xff0c;“系统设置”&#xff0c;“安全级别”&#xff1b;然后在“安全级别配置”对…

JDK源码解析之 Java.lang.Double

Double类是原始类型double的包装类&#xff0c;它包含若干有效处理double值的方法&#xff0c;如将其转换为字符串表示形式&#xff0c;反之亦然。Double类的对象可以包含一个double值。 Double类包装原始类型的值 double中的对象。类型的对象 Double包含一个类型为的字段 doub…

网页搜索帮助-禁止搜索引擎收录的方法

什么是robots.txt文件?搜索引擎使用spider程序自动访问互联网上的网页并获取网页信息。spider在访问一个网站时&#xff0c;会首先会检查该网站的根域下是否有一个叫做robots.txt的纯文本文件。您可以在您的网站中创建一个纯文本文件robots.txt&#xff0c;在文件中声明该网站…

JDK源码解析之 Java.lang.Float

Float类是原始类型float的包装类&#xff0c;它包含若干有效处理浮点值的方法&#xff0c;如将其转换为字符串表示形式&#xff0c;反之亦然。Float类的一个对象可以包含一个浮点值 一、类定义 public final class Float extends Number implements Comparable<Float> {…

FTP两种工作模式:主动模式(Active FTP)和被动模式(Passive FTP)

在主动模式下&#xff0c;FTP客户端随机开启一个大于1024的端口N向服务器的21号端口发起连接&#xff0c;然后开放N1号端口进行监听&#xff0c;并向服务器发出PORT N 1命令。服务器接收到命令后&#xff0c;会用其本地的FTP数据端口&#xff08;通常是20&#xff09;来连接客户…

JDK源码解析之 java.lang.Integer

teger 基本数据类型int 的包装类 Integer 类型的对象包含一个 int 类型的字段 一、类定义 public final class Integer extends Number implements Comparable<Integer>{}类被声明为final的,表示不能被继承;继承了Number抽象类,可以用于数字类型的一系列转换;实现了Comp…

Loadrunner的基本概念

1)vuser_init(only one &#xff0c;重复执行脚本的时候&#xff0c;此部分只会执行一次 ) %G< rJc*P 2)action( 一个或者多个 , 重复执行脚本的时候&#xff0c;只有该部分会按重复的次数重复执行 ) z*Xfjy(Mj 3)vuser_end(only one, 重复执行脚本的时候&#xff0c;此…

JDK源码解析之 java.lang.Long

Long 与Integer 是数值类型中使用频率最高的两个,也是提供支持方法最多的两个 他们提供出来的方法功能也是高度的相似 一、类定义 public final class Long extends Number implements Comparable<Long> {}类被声明为final的,表示不能被继承;继承了Number抽象类,可以用于…

sed教程入门与实例练习(一)

UNIX 世界中有很多文本编辑器可供我们选择。思考一下 — vi、emacs 和 jed 以及很多其它工具都会浮现在脑海中。我们都有自己已逐渐了解并且喜爱的编辑器&#xff08;以及我们喜爱的组合键&#xff09;。有了可信赖的编辑器&#xff0c;我们可以轻松处理任何数量与 UNIX 有关的…

JDK源码解析之 Java.lang.Short

Short类是基本类型short 的包装类&#xff0c;它包含几种有效处理短值的方法&#xff0c;如将其转换为字符串表示形式&#xff0c;反之亦然。Short类的对象可以包含单个短值。 一、类定义 public final class Short extends Number implements Comparable<Short> {}类被…

sed教程入门与实例练习(二)

让我们看一下 sed 最有用的命令之一&#xff0c;替换命令。使用该命令&#xff0c;可以将特定字符串或匹配的规则表达式用另一个字符串替换。下面是该命令最基本用法的示例&#xff1a; $ sed -e ’s/foo/bar/’ myfile.txt上面的命令将 myfile.txt 中每行第一次出现的 ‘foo’…

Oracle GoldenGate简介

一、什么是Oracle GoldenGate&#xff1f; Oracle GoldenGate是用于实时数据集成和复制的综合软件包。它支持高可用性解决方案&#xff0c;实时数据集成&#xff0c;事务性更改数据捕获&#xff0c;数据复制&#xff0c;转换以及运营和分析企业系统之间的验证。 使用Oracle G…