解决StringBuffer和StringBuilder的扩容问题
StringBuffer和StringBuilder的扩容
以前对StringBuffer/StringBuilder只是初浅的了解,只是知道StringBuffer是线程安全的,Stringbuilder是非线程安全的。并且字符串长度是可变的。
具体是怎么变没有去深入了解,今天看了一下源码,恍然大悟
来看一下源码,在没有传参的情况下默认初始容量是16。
文章图片
有参数的情况下,初始容量是16+字符串的长度,并且是用append()方法追加的字符。
文章图片
到这里就有疑问那,那这个字符串的长度是多少呢。是它本身的长度还是16+它自身的长度,来接着往下看。一路追寻append()方法终于找到答案了。
【解决StringBuffer和StringBuilder的扩容问题】注意,这个len居然是String自身的长度,现在明白了吧。其实平时咱们也在用str.length();方法就是没注意它的长度是怎么来的。
文章图片
忽然又想到一个问题,那要是在追加字符串的时候长度比16大怎么办,我们看到有个ensureCapacityInternal()的方法,追进去看看,然后发现它是这么扩容的
int newCapacity = (value.length << 1) + 2;
增加为自身长度的一倍然后再加2;这个时候如果还是放不下,那就直接扩容到它需要的长度
newCapacity = minCapacity;
文章图片
文章图片
文章图片
StringBuilder扩容规则 StringBuilder默认的创建的时候开辟的char数组的大小
StringBuilder() default 16StringBuilder("Str") default Str.length()+16
StringBuilder sb扩容的规则:
当调用sb.append()的时候每次都会对当前容量进行判断
文章图片
确定下需要的最小的容量(已经存储的数据长度+准备存储的数据的长度)是否大于存储的char数组的长度,如果大于就在newCapacity扩容
文章图片
判断扩容当前char数组长度的2倍+2的长度是否满足扩容需求,不满足设置为存储的数据长度+准备存储的数据的长度,判断append扩容是否超过MAX_ARRAY_SIZE(Integer.MAX_VALUE - 8),如果是抛出异常OutOfMemoryError
文章图片
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
推荐阅读
- 急于表达——往往欲速则不达
- parallels|parallels desktop 解决网络初始化失败问题
- 第三节|第三节 快乐和幸福(12)
- 20170612时间和注意力开销记录
- 2.6|2.6 Photoshop操作步骤的撤消和重做 [Ps教程]
- 对称加密和非对称加密的区别
- 眼光要放高远
- 考研英语阅读终极解决方案——阅读理解如何巧拿高分
- 樱花雨
- 前任