String内部是一个private final char value[];
也就意味着每次调用的各种处理方法,返回的字符串都是一个新的,性能上,显然....
所以,对于可变字符序列的需求是很明确的
类的层次结构设计,有的时候是自顶而下
有的时候是总结归纳,然后抽象出来一个新的类,这很正常
尽管StringBuffer 要比StringBuilder 和 AbstractStringBuilder要早得多了
但是StringBuffer 现在也继承了这个类
所以说,抛开发展历史的过程不说,直接从类的层级结构设计的角度看的话
你可以认为AbstractStringBuilder 就是对于可变字符序列的这一概念的描述
他提供了可变字符序列的一个基本协议约定,也就是基本的功能方法
作为一个抽象类, 并且也提供了一部分默认的实现
StringBuffer和StringBuilder都是可变的字符序列,所以他们都实现了AbstractStringBuilder
|
属性简介
内部是一个char[] value,不再是final的了,也就意味着可变
他实现了CharSequence接口意味着他是一个字符序列
实现了Appendable接口,意味着他遵循了追加相关的协议
|
内部使用char[] value进行数据存储 , 这个char[] value 是核心
他是有容量大小的,因为数组必然有长度
如果长度没有超出此容量,就无需分配新的内部缓冲区数组
如果内部缓冲区溢出,则此容量自动增大
使用count 记录已经使用的字符个数
还可以通过有参数的构造方法进行初始化设置value这个字符数组的大小
当然构造方法不是给你用的,是给子类用的
既然本质是一个char[] 字符数组,所以可以说所有的操作都是对于数组的操作
那么对于一个字符序列,有哪些方法是刚需呢?
也无外乎添加元素/删除元素/更新元素/获取元素 这几种
添加有可能是插入中间或者在最后追加
我们知道,数组是顺序存储
所以对于插入这种操作必然会出现大量的元素移动情况
|
属性获取
既然是内部维护了字节数组
必然这个数组本身属性,长度 ,使用个数的获取,以及数组的扩大也有相对应的方法可以使用
length() 获取实际数据的个数
capacity() 数组的大小 所以是容量
|
|
public void setLength(int newLength) | 设置为指定长度
如果 newLength 参数小于当前长度
则长度将更改为指定的长度, 截断,数据不变
如果 newLength 参数大于或等于当前长度
则将追加有效的 null 字符 ('\u0000'),使长度满足 newLength 参数
|
public void ensureCapacity(int minimumCapacity) |
确保容量至少等于指定的最小值
如果 minimumCapacity 参数为非正数,则此方法不执行任何操作并返回
参数大于当前容量才会执行扩展
新容量的大小应大于:minimumCapacity 参数 并且大于 旧容量的两倍加 2
|
public void trimToSize() | 尝试缩减空间 如果实际使用的个数小于容量,那么进行缩减 |
添加方法
就像我们刚才说的那样,添加元素,分为尾部追加元素和中间插入元素
添加元素的方法也分为了两大阵营
AbstractStringBuilder append(Object obj) | AbstractStringBuilder insert(int offset, Object obj) |
AbstractStringBuilder append(boolean b) | AbstractStringBuilder insert(int offset, boolean b) |
AbstractStringBuilder append(char c) | AbstractStringBuilder insert(int offset, char c) |
AbstractStringBuilder append(int i) | AbstractStringBuilder insert(int offset, int i) |
AbstractStringBuilder append(long l) | AbstractStringBuilder insert(int offset, long l) |
AbstractStringBuilder append(float f) | AbstractStringBuilder insert(int offset, float f) |
AbstractStringBuilder append(double d) | AbstractStringBuilder insert(int offset, double d) |
AbstractStringBuilder append(CharSequence s) | AbstractStringBuilder insert(int dstOffset, CharSequence s) |
AbstractStringBuilder append(CharSequence s, int start, int end) |
AbstractStringBuilder insert(int dstOffset, CharSequence s,int start, int end)
|
AbstractStringBuilder append(char[] str) | AbstractStringBuilder insert(int offset, char[] str) |
AbstractStringBuilder append(char str[], int offset, int len) |
AbstractStringBuilder insert(int index, char[] str, int offset, int len)
|
AbstractStringBuilder append(String str) | AbstractStringBuilder insert(int offset, String str) |
AbstractStringBuilder append(StringBuffer sb) | |
AbstractStringBuilder appendCodePoint(int codePoint) |
从上表可以看得出来,给各种数据类型都提供了append和insert方法 |
对于insert 是插入,既然是插入,那么就需要指定位置 所以与append对应的方法的方法签名上,都多了一个索引 |
所有的方法的返回类型都是AbstractStringBuilder 其实都是 return this; 因为他是可变的,所以变化直接体现在了this中,所以返回this就好了 |
appendCodePoint(int codePoint) 与 append(char c) 可以说是一样的 但是对于辅助平面显然又不一样,char不支持辅助平面 |
获取方法
获取代码点
代码点相关的五个方法
charAt(int) / codePointAt(int) / codePointBefore(int) / codePointCount(int, int) / offsetByCodePoints(int, int)
他们与String中的是一模一样的,代码也是一样的(就有个变量名变动)
|
复制
getChars(int, int, char[], int)
将字符从此序列复制到目标字符数组 dst 与String中的方法也几乎一致
|
索引下标
int indexOf(String str)
int indexOf(String str, int fromIndex)
|
第一次出现的指定子字符串在该字符串中的索引 可以指定索引 指定索引就从索引处开始查找匹配 满足的条件为startsWith true (并且在范围内 如果有设置) |
int lastIndexOf(String str)
int lastIndexOf(String str, int fromIndex)
|
返回最右边出现的指定子字符串在此字符串中的索引 也就是最后一个 可以指定索引 指定索引就从索引处 反向匹配 满足的条件也是startsWith true (并且在范围内 如果有设置) |
获取子串
public
String substring(int start, int end)
|
根据索引返回子串 |
public String substring(int start) | substring(int start, int end)的简化方法 指定开始位置,默认结束位置为最后 |
public CharSequence subSequence(int start, int end) |
为了实现CharSequence方法 内部调用的substring |
更新方法
更新方法比较少,因为是数组 数组的访问按照下标进行设置就好了 还提供了替换的功能,也算是更新操作 |
AbstractStringBuilder replace(int start, int end, String str) 使用str替换对象中从start 开始到end结束的这一段 |
删除方法
AbstractStringBuilder delete(int start, int end) | 删除指定范围的char |
AbstractStringBuilder deleteCharAt(int index) | 删除某个位置的char |
其他方法
reverse() 按照字符进行翻转 |
将此字符序列用其反转形式取代 就是翻转每一个char,注意可不是翻转比特位,也不是翻转字节,反转的是代码单元 不过对于辅助平面的字符的代码点,他们会按照字符进行翻转,也就是高代理低代理顺序不会改变 |
执行操作前未成对的低代理项和高代理项将成为代理项对 例如,反转 "\uDC00\uD800" 将生成有效的代理项对 "\uD800\uDC00" |
总结
如同我们上面说的,AbstractStringBuilder就是 可变 字符序列的一个纲领
它规定了可变字符序列应该有的行为
比如 添加字符/删除字符/更新字符/获取字符
因为可变,所以对于可变的支持,自然是必不可少的
另外,他作为String在很多方面的一个替代,必然也是提供了String的一些功能方法
否则与String API 变化巨大 也是毫无意义
因为毕竟本身就是为了描述字符序列
所以对于AbstractStringBuilder 只需要彻底理解了他作为 可变字符序列的标准接口即可