StringBuffer 存储和操作字符串
它所继承实现的类和接口
public final class StringBuffer extends AbstractStringBuilder implements java.io.Serializable, CharSequence
构造一个其中不带字符的字符串缓冲区,初始容量为 16 个字符。
StringBuffer.class
public StringBuffer() { super(16); }
AbstractStringBuilder.class
char value[];//存内容的字符数组
,StringBuffer中未定义新的变量,一直在操作父类定义的变量
AbstractStringBuilder(int capacity) {
value = new char[capacity];//在这个例子中是16
}
构造一个不带字符,但具有指定初始容量的字符串缓冲区。
构造一个字符串缓冲区,并将其内容初始化为指定的字符串内容。该字符串的初始容量为 16 加上字符串参数的长度。
StringBuffer.class
public StringBuffer(String str) { //这个执行父类的带参构造函数AbstractStringBuilder(int capacity) super(str.length() + 16); append(str); } public synchronized StringBuffer append(String str) { //执行父类的append(str) super.append(str); return this; }
AbstractStringBuilder.class
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; } char value[];//存内容的字符数组 //扩张容量 void expandCapacity(int minimumCapacity) { //newCapacity 新容量,默认+1*2 int newCapacity = (value.length + 1) * 2; if (newCapacity < 0) { //数字超过了Integer的最大值 newCapacity = Integer.MAX_VALUE; } else if (minimumCapacity > newCapacity) { //默认扩大的容量还是不足存新的字符 newCapacity = minimumCapacity; } //将旧的值剪切到新的字符数组。 value = Arrays.copyOf(value, newCapacity); }
append的步骤类似,
1.先判断是否为null,null就将str="null".
2.再判断str的长度是否为0,为0直接返回。
3.判断value[]字符数组的剩余空间够不够。
4.不够的话new一个char[] 空间为(原来+1)*2,并且判断时候超出Integer的最大值。然后System.arraycopy,将旧的字符数组复制到新的数组中
5.str.getChars,将str写入value[] 中。调用的也是System.arraycopy
先执行String.valueOf(obj),然后执行append(str)
第五步不同,调用StringBuffer的getChars方法,内部一样为System.arraycopy.
1.直接相加计算新的字符长度,
2.长度不够扩张
3.System.arraycopy复制
1.判断s对象是String 或者StringBuffer的子类,分别调用append(str)或者append(sb)
2.如果不是,则利用value[count++] = s.charAt(i);
value[count++] = 't';
value[count++] = 'r';
value[count++] = 'u';
value[count++] = 'e';
int appendedLength = (i < 0) ? stringSizeOfInt(-i) + 1 : stringSizeOfInt(i);//计算i的长度
final static int [] sizeTable = { 9, 99, 999, 9999, 99999, 999999, 9999999,99999999, 999999999, Integer.MAX_VALUE }; static int stringSizeOfInt(int x) { for (int i=0; ; i++) if (x <= sizeTable[i]) return i+1; }
上面的代码计算出数字i的长度
然后利用Integer.getChars将i复制进value[]中,这个方法也很特别,以后再看。
int appendedLength = (l < 0) ? stringSizeOfLong(-l) + 1 : stringSizeOfLong(l);
static int stringSizeOfLong(long x) { long p = 10; for (int i=1; i<19; i++) { if (x < p) return i; p = 10*p; } return 19; }
这个是计算long的长度的方法,利用Long.getChars将内容复制入新字符数组中
移除此序列的子字符串中的字符。系统直接将后面的字符用 System.arrayCopy复制面搜索的部分大删除起始点。删除掉了start ,end未删除。
直接删除一个字符,相当于delete(index,index+1)
先看容量够不够,end后的内容利用System.arraCopy空出str的长度,然后将str复制进字符数组、
执行return new String(value, 0, count);
好了,就看到这吧。