爆款云主机2核4G限时秒杀,88元/年起!
查看详情

活动

天翼云最新优惠活动,涵盖免费试用,产品折扣等,助您降本增效!
热门活动
  • 618智算钜惠季 爆款云主机2核4G限时秒杀,88元/年起!
  • 免费体验DeepSeek,上天翼云息壤 NEW 新老用户均可免费体验2500万Tokens,限时两周
  • 云上钜惠 HOT 爆款云主机全场特惠,更有万元锦鲤券等你来领!
  • 算力套餐 HOT 让算力触手可及
  • 天翼云脑AOne NEW 连接、保护、办公,All-in-One!
  • 中小企业应用上云专场 产品组合下单即享折上9折起,助力企业快速上云
  • 息壤高校钜惠活动 NEW 天翼云息壤杯高校AI大赛,数款产品享受线上订购超值特惠
  • 天翼云电脑专场 HOT 移动办公新选择,爆款4核8G畅享1年3.5折起,快来抢购!
  • 天翼云奖励推广计划 加入成为云推官,推荐新用户注册下单得现金奖励
免费活动
  • 免费试用中心 HOT 多款云产品免费试用,快来开启云上之旅
  • 天翼云用户体验官 NEW 您的洞察,重塑科技边界

智算服务

打造统一的产品能力,实现算网调度、训练推理、技术架构、资源管理一体化智算服务
智算云(DeepSeek专区)
科研助手
  • 算力商城
  • 应用商城
  • 开发机
  • 并行计算
算力互联调度平台
  • 应用市场
  • 算力市场
  • 算力调度推荐
一站式智算服务平台
  • 模型广场
  • 体验中心
  • 服务接入
智算一体机
  • 智算一体机
大模型
  • DeepSeek-R1-昇腾版(671B)
  • DeepSeek-R1-英伟达版(671B)
  • DeepSeek-V3-昇腾版(671B)
  • DeepSeek-R1-Distill-Llama-70B
  • DeepSeek-R1-Distill-Qwen-32B
  • Qwen2-72B-Instruct
  • StableDiffusion-V2.1
  • TeleChat-12B

应用商城

天翼云精选行业优秀合作伙伴及千余款商品,提供一站式云上应用服务
进入甄选商城进入云市场创新解决方案
办公协同
  • WPS云文档
  • 安全邮箱
  • EMM手机管家
  • 智能商业平台
财务管理
  • 工资条
  • 税务风控云
企业应用
  • 翼信息化运维服务
  • 翼视频云归档解决方案
工业能源
  • 智慧工厂_生产流程管理解决方案
  • 智慧工地
建站工具
  • SSL证书
  • 新域名服务
网络工具
  • 翼云加速
灾备迁移
  • 云管家2.0
  • 翼备份
资源管理
  • 全栈混合云敏捷版(软件)
  • 全栈混合云敏捷版(一体机)
行业应用
  • 翼电子教室
  • 翼智慧显示一体化解决方案

合作伙伴

天翼云携手合作伙伴,共创云上生态,合作共赢
天翼云生态合作中心
  • 天翼云生态合作中心
天翼云渠道合作伙伴
  • 天翼云代理渠道合作伙伴
天翼云服务合作伙伴
  • 天翼云集成商交付能力认证
天翼云应用合作伙伴
  • 天翼云云市场合作伙伴
  • 天翼云甄选商城合作伙伴
天翼云技术合作伙伴
  • 天翼云OpenAPI中心
  • 天翼云EasyCoding平台
天翼云培训认证
  • 天翼云学堂
  • 天翼云市场商学院
天翼云合作计划
  • 云汇计划
天翼云东升计划
  • 适配中心
  • 东升计划
  • 适配互认证

开发者

开发者相关功能入口汇聚
技术社区
  • 专栏文章
  • 互动问答
  • 技术视频
资源与工具
  • OpenAPI中心
开放能力
  • EasyCoding敏捷开发平台
培训与认证
  • 天翼云学堂
  • 天翼云认证
魔乐社区
  • 魔乐社区

支持与服务

为您提供全方位支持与服务,全流程技术保障,助您轻松上云,安全无忧
文档与工具
  • 文档中心
  • 新手上云
  • 自助服务
  • OpenAPI中心
定价
  • 价格计算器
  • 定价策略
基础服务
  • 售前咨询
  • 在线支持
  • 在线支持
  • 工单服务
  • 建议与反馈
  • 用户体验官
  • 服务保障
  • 客户公告
  • 会员中心
增值服务
  • 红心服务
  • 首保服务
  • 客户支持计划
  • 专家技术服务
  • 备案管家

了解天翼云

天翼云秉承央企使命,致力于成为数字经济主力军,投身科技强国伟大事业,为用户提供安全、普惠云服务
品牌介绍
  • 关于天翼云
  • 智算云
  • 天翼云4.0
  • 新闻资讯
  • 天翼云APP
基础设施
  • 全球基础设施
  • 信任中心
最佳实践
  • 精选案例
  • 超级探访
  • 云杂志
  • 分析师和白皮书
  • 天翼云·创新直播间
市场活动
  • 2025智能云生态大会
  • 2024智算云生态大会
  • 2023云生态大会
  • 2022云生态大会
  • 天翼云中国行
天翼云
  • 活动
  • 智算服务
  • 产品
  • 解决方案
  • 应用商城
  • 合作伙伴
  • 开发者
  • 支持与服务
  • 了解天翼云
      • 文档
      • 控制中心
      • 备案
      • 管理中心

      StringBuffer源码分析之 append 方法

      首页 知识中心 软件开发 文章详情页

      StringBuffer源码分析之 append 方法

      2024-04-17 02:21:24 阅读次数:53

      StringBuffer

      StringBuffer 这个类是我们日常开发中经常会使用的一个字符串操作类,该类提供了非常多的关于字符串操作相关的类,尤其是 append 方法更为常用。 

      1 目标

      本次源码分析的目标是深入了解 StringBuffer类中 append 方法的实现机制。

      2 分析方法

      首先编写测试代码,然后利用 Intellij Idea 的单步调试功能,逐步的分析其实现思路。

      测试代码如下:

      StringBuffer stringBuffer = new StringBuffer(); //断点
      
      stringBuffer.append("hello");
      stringBuffer.append("hello11");
      
      stringBuffer.append("hello22");
      
      String nullStr = null;
      stringBuffer.append(nullStr);
      
      
      3 分析流程

      点击调试按钮,开始分析流程。

      3.1 构造函数

      首先进行的是构造函数的分析,点击 F7进入构造函数实现。

      此时需要注意的是,当我们点击 F7发现Idea 无响应并未进入构造函数内部实现。这是为什么?

      Idea 的 F7 (step into)默认是不进入JDK 的类实现,而 StringBuffer 类正是 JDK 中 lang 包下的类,因此点击 F7并未跳到内部实现。此时应该选择Shift+F7(force step into) 按键。 

      /**
       * Constructs a string buffer with no characters in it and an
       * initial capacity of 16 characters.
       */
      public StringBuffer() {
          super(16);
      }
      

      点击进入后,我们看到的是以上代码,表示调用父类的构造函数,并附上参数16。StringBuffer 的父类是谁?这个数字16又是什么意思呢?

      我们再按 Shift + F7进入查看做进一步分析。

      /**
       * Creates an AbstractStringBuilder of the specified capacity.
       */
      AbstractStringBuilder(int capacity) {
          value = new char[capacity];
      }

      通过上述代码我们知道,StringBuffer 的父类是 AbstractStringBuilder,这是一个抽象类。其构造函数初始化了一个默认大小的字符数组,而这个字符数组的大小正是传进来的参数。

      通过字符数组来保存字符串信息,为什么默认大小为16,如果字符串超过16,超过了字符数组的大小了怎么办?

      我们希望通过后续的分析能够解决上面提出的这个问题。 

      因此,我们了解到,StringBuffer 的构造函数本质是调用了父类 AbstractStringBuilder 类的构造函数,该构造函数初始化了一个默认大小为16的字符数组。

      3.2 append()方法

      接下来我们分析 append 方法的实现机制。

      @Override
      public synchronized StringBuffer append(String str) {
          toStringCache = null;
          super.append(str);
          return this;
      }
      

      从上述代码可以看到,直接调用了父类 AbstractStringBuilder 类的 append 方法。

      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;
      }

      首先判断追加的字符串是否为 null,如果为 null 则执行 appendNull()方法。下一节我们再分析这个方法。

      根据上一节测试代码编写的:stringBuffer.append("hello");

      我们追加的是一个字符串"hello",并非 null 值。

      获取追加字符串的长度 len值为5。

      下面将进入 ensureCapacityInternal ()方法,该方法的参数为 count+len = 0 + 5 = 5.

      这个 count属性是什么意思呢?

      将鼠标放在 count 上面,点击 Ctrl+B 进入到该属性的定义:

      /**
       * The count is the number of characters used.
       */
      int count;

       

      从代码的注释我们可以看到,count 表示的是已经使用的字符数量。从§3.1节构造函数我们知道这些字符串都是存储在一个字符数组中,而 count 指的就是这个字符数组已经使用了多少个。

      由于我们是第一次执行 append 方法,此前没有追加任何的字符,因此此时 count 为0,当我们追加完成后,此时 count 的值就要更新为5,表示此时的字符数组中已经有5个字符了。

      所以 ensureCapacityInternal()方法的参数指的是已经使用的字符数量+将要使用的字符数量,即字符数组的最小容量大小。方法分析见§3.3节,该方法确定了新字符数组的容量,并初始化新字符数组,将原有字符数组内容复制到新字符数组中。

      str.getChars()方法分析见§3.5节,主要完成将追加的字符串复制到字符数组中。

      str.getChars(0, len, value, count);

      将追加的字符串str的0-len位置的字符复制到 字符数组 value 的起始位置 count 处。

       因此,append 方法主要的工作是:获得要追加的字符串的长度,判断当前字符数组是否能够存储追加的字符串,如果容量不够则确定新的字符数组的容量,申请新的字符数组,将以前字符数组的内容复制到新字符数组。最后将要追加的字符串,复制到新字符数组中。

      3.3 ensureCapacityInternal()方法

       按住 Shift+F7进入该方法的实现代码。 

      /**
       * For positive values of {@code minimumCapacity}, this method
       * behaves like {@code ensureCapacity}, however it is never
       * synchronized.
       * If {@code minimumCapacity} is non positive due to numeric
       * overflow, this method throws {@code OutOfMemoryError}.
       */
      private void ensureCapacityInternal(int minimumCapacity) {
          // overflow-conscious code
          if (minimumCapacity - value.length > 0) {
              value = Arrays.copyOf(value,
                      newCapacity(minimumCapacity));
          }
      }

       

      通过§3.1节构造函数的分析,我们了解到 StringBuffer 的底层是使用字符数组来存储这些字符串的,而且默认的大小是16,一旦这个字符数组用完了,就得重新分配新的字符数组,并将以前的字符数组内容复制到新的字符数组中。

      minimumCapacity指的就是当前字符串的最小容量,如果这个容量比当前字符数组的容量要大,则需要重新申请新的字符数组,并将以前字符数组的内容复制到新的字符数组中。

      那么新的字符数组容量是多少呢?

      答案就在 newCapacity(minimumCapacity)方法中,见§3.4节。

      有了新的字符数组了以后,接下来就需要将以前的字符数组的内容复制到新的字符数组,通过 Arrays.copyOf()方法实现。

      因此,ensureCapacityInternal()完成的工作主要是确定新的字符数组的大小并将旧字符数组的内容复制到新字符数组中。

      3.4 newCapacity()方法

       按住 Shift+F7进入该方法的实现。

      /**
       * Returns a capacity at least as large as the given minimum capacity.
       * Returns the current capacity increased by the same amount + 2 if
       * that suffices.
       * Will not return a capacity greater than {@code MAX_ARRAY_SIZE}
       * unless the given minimum capacity is greater than that.
       *
       * @param  minCapacity the desired minimum capacity
       * @throws OutOfMemoryError if minCapacity is less than zero or
       *         greater than Integer.MAX_VALUE
       */
      private int newCapacity(int minCapacity) {
          // overflow-conscious code
          int newCapacity = (value.length << 1) + 2;
          if (newCapacity - minCapacity < 0) {
              newCapacity = minCapacity;
          }
          return (newCapacity <= 0 || MAX_ARRAY_SIZE - newCapacity < 0)
              ? hugeCapacity(minCapacity)
              : newCapacity;
      }
      

       

      默认的 newCapacity 大小是原有的字符数组大小左移一位加上2,即2*oldCapacity+2,将原有的字符数组扩大一倍再加上2。为什么是这样的一种算法呢?直接左移一位不就可以了吗?为什么还要加2?

      这里面的newCapacity 和 minCapacity 两个变量容易产生混淆,其中 newCapacity 指的是字符数组新的容量大小,而 minCapacity 指的是当前要存储字符串而需要的最小容量。因此要想能够存储当前字符串,就必须保证 newCapacity >= minCapacity。

      所以上述源码中加了一个判断newCapacity 是否大于 minCapacity,如果不是则 newCapacity 的大小直接设置为 minCapacity。

      最后返回的时候,还加上了相关判断信息,当 newCapacity 超过了当前数组的最大值的时候,执行 hugeCapacity()方法。 

      3.5 str.getChars()方法

       按住 Shift+F7进入该方法的实现代码:

      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);
      }
      

      getChars()为 String 类的方法,通过调用 System.arraycopy()系统方法完成将当前字符串的 scrBegin ~ srcEnd 复制到字符数组的 dstBegin 位置。

      3.6 appendNull()方法

      String nullStr = null;
      stringBuffer.append(nullStr);

      当我们追加的是一个 null 串的时候,StringBuffer 是如何处理的。 

      private AbstractStringBuilder appendNull() {
          int c = count;
          ensureCapacityInternal(c + 4);
          final char[] value = this.value;
          value[c++] = 'n';
          value[c++] = 'u';
          value[c++] = 'l';
          value[c++] = 'l';
          count = c;
          return this;
      }

      这是一个私有的方法。首先确保容量足够,其次我们看到所谓的 null 本质就是追加了'null'这样的四个字符到字符数组中。 

      4 总结

      本文分析了 StringBuffer 类的 append 方法,通过分析我们知道append方法的所有工作都是由父类 AbstractStringBuilder 完成的。基本的思路是检查当前字符数组的容量是否足够,如果不够,则申请新的字符数组,然后将原有字符数组的内容复制到新的字符数组。最后将追加的字符串复制到新的字符数组后面,从而完成追加操作。

      如果让你来设计一个类来完成字符串的不断追加操作,你会怎么设计呢?

      可能大部分同学想到的都是每次申请一个新的字符数组,将原有数组的内容复制到新数组,最后将追加的字符串复制到新数组后面。

      这种实现方式最大的问题就是效率。不停的进行数组的复制操作导致效率非常低下,因此StringBuffer 提出的思路是每次我多申请一些字符数组,当容量不够的时候,申请原有容量2倍+2的容量,而不仅仅是满足 minCapacity 最小容量的大小。这就是提升效率的一种方式,这种设计方式在很多场景都有应用。

      通过源码分析,我们深入了解一个类的内部实现机制,使得我们今后会更加高效的使用这个类。另外我们还会学习到一些 Java 编程的技巧和一些设计思路。

      版权声明:本文内容来自第三方投稿或授权转载,原文地址:https://blog.51cto.com/where2go/2956545,作者:算法与编程之美,版权归原作者所有。本网站转在其作品的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如因作品内容、版权等问题需要同本网站联系,请发邮件至ctyunbbs@chinatelecom.cn沟通。

      上一篇:线程--线程创建与终止

      下一篇:我爱java系列---【springboot集成redis】

      相关文章

      2025-03-10 09:53:00

      Java学习路线-17:Java基础类库StringBuffer、AutoCloseable、Runtime、System

      Java学习路线-17:Java基础类库StringBuffer、AutoCloseable、Runtime、System

      2025-03-10 09:53:00
      JDK , String , StringBuffer , 使用 , 接口 , 示例
      2025-02-12 09:27:42

      初始Java篇(JavaSE基础语法)(8)认识String类(下)

      初始Java篇(JavaSE基础语法)(8)认识String类(下)

      2025-02-12 09:27:42
      int , String , StringBuffer , 修改 , 字符 , 字符串 , 遍历
      2024-12-13 06:54:00

      java中的String、StringBuffer和StringBuilder的详细分析

      java中的String、StringBuffer和StringBuilder的详细分析

      2024-12-13 06:54:00
      String , StringBuffer , 字符串 , 源码 , 线程
      2024-06-18 07:17:34

      面试官请不要再问我StringBuffer与StringBuilder的区别了,源码讲你你听

      面试官请不要再问我StringBuffer与StringBuilder的区别了,源码讲你你听

      2024-06-18 07:17:34
      StringBuffer , StringBuilder
      2024-06-05 09:49:00

      【设计模式】建造者模式——建造者模式在JDK源码中的应用

      因为建造者模式一般由静态内部类实现,所以在JDK源码中搜索“public static class Builder”可以得到很多结果,但大多数与应用层程序员关系不大,我这里挑几个常用的和建造者模式相关的类,与大家讨论一下。

      2024-06-05 09:49:00
      Java , StringBuffer , StringBuilder
      2024-05-16 08:23:35

      Java 中 StringBuffer 线程安全的小介绍

      因为最近在研究线程安全和相关的可变对象和不可变对象的内容。 查看了下 StringBuffer 的源代码。

      2024-05-16 08:23:35
      Java , StringBuffer
      2024-05-13 07:32:14

      判断是否是回文字符串(Java实现)

      “回文数”就是正读倒读都一样的整数。如奇数个数字:98789,这个数字正读是98789 倒读也是98789。偶数个数字3223也是回文数。字母 abcba 也是回文。

      2024-05-13 07:32:14
      StringBuffer , 字符串
      2023-08-03 08:24:01

      StringBuffer类和StringBuilder类

      StringBuffer类和StringBuilder类

      2023-08-03 08:24:01
      java , StringBuffer , StringBuilder
      2023-06-08 06:23:00

      StringBuffer和StringBuilder类

      1. 基本介绍  474java.lang.StringBuffer代表可变的字符序列,可以对字符串内容进行增删。很多方法与String相同,但StringBuffer是可变长度的。StringBuffer是一个容器。代码在com.stul

      2023-06-08 06:23:00
      StringBuffer , StringBuilder
      2023-05-31 08:45:09

      String、StringBuffer和StringBuilder类的区别以及StringBuffer的常用方法 StringBuffer的capacity容量扩容机制

      String、StringBuffer和StringBuilder类的区别以及StringBuffer的常用方法 StringBuffer的capacity容量扩容机制

      2023-05-31 08:45:09
      String , StringBuffer
      查看更多
      推荐标签

      作者介绍

      天翼云小翼
      天翼云用户

      文章

      33561

      阅读量

      5237672

      查看更多

      最新文章

      Java学习路线-17:Java基础类库StringBuffer、AutoCloseable、Runtime、System

      2025-03-10 09:53:00

      初始Java篇(JavaSE基础语法)(8)认识String类(下)

      2025-02-12 09:27:42

      java中的String、StringBuffer和StringBuilder的详细分析

      2024-12-13 06:54:00

      面试官请不要再问我StringBuffer与StringBuilder的区别了,源码讲你你听

      2024-06-18 07:17:34

      【设计模式】建造者模式——建造者模式在JDK源码中的应用

      2024-06-05 09:49:00

      Java 中 StringBuffer 线程安全的小介绍

      2024-05-16 08:23:35

      查看更多

      热门文章

      Java 中 StringBuffer 线程安全的小介绍

      2024-05-16 08:23:35

      【设计模式】建造者模式——建造者模式在JDK源码中的应用

      2024-06-05 09:49:00

      面试官请不要再问我StringBuffer与StringBuilder的区别了,源码讲你你听

      2024-06-18 07:17:34

      判断是否是回文字符串(Java实现)

      2024-05-13 07:32:14

      java中的String、StringBuffer和StringBuilder的详细分析

      2024-12-13 06:54:00

      初始Java篇(JavaSE基础语法)(8)认识String类(下)

      2025-02-12 09:27:42

      查看更多

      热门标签

      java Java python 编程开发 代码 开发语言 算法 线程 Python html 数组 C++ 元素 javascript c++
      查看更多

      相关产品

      弹性云主机

      随时自助获取、弹性伸缩的云服务器资源

      天翼云电脑(公众版)

      便捷、安全、高效的云电脑服务

      对象存储

      高品质、低成本的云上存储服务

      云硬盘

      为云上计算资源提供持久性块存储

      查看更多

      随机文章

      判断是否是回文字符串(Java实现)

      java中的String、StringBuffer和StringBuilder的详细分析

      Java 中 StringBuffer 线程安全的小介绍

      面试官请不要再问我StringBuffer与StringBuilder的区别了,源码讲你你听

      【设计模式】建造者模式——建造者模式在JDK源码中的应用

      初始Java篇(JavaSE基础语法)(8)认识String类(下)

      • 7*24小时售后
      • 无忧退款
      • 免费备案
      • 专家服务
      售前咨询热线
      400-810-9889转1
      关注天翼云
      • 旗舰店
      • 天翼云APP
      • 天翼云微信公众号
      服务与支持
      • 备案中心
      • 售前咨询
      • 智能客服
      • 自助服务
      • 工单管理
      • 客户公告
      • 涉诈举报
      账户管理
      • 管理中心
      • 订单管理
      • 余额管理
      • 发票管理
      • 充值汇款
      • 续费管理
      快速入口
      • 天翼云旗舰店
      • 文档中心
      • 最新活动
      • 免费试用
      • 信任中心
      • 天翼云学堂
      云网生态
      • 甄选商城
      • 渠道合作
      • 云市场合作
      了解天翼云
      • 关于天翼云
      • 天翼云APP
      • 服务案例
      • 新闻资讯
      • 联系我们
      热门产品
      • 云电脑
      • 弹性云主机
      • 云电脑政企版
      • 天翼云手机
      • 云数据库
      • 对象存储
      • 云硬盘
      • Web应用防火墙
      • 服务器安全卫士
      • CDN加速
      热门推荐
      • 云服务备份
      • 边缘安全加速平台
      • 全站加速
      • 安全加速
      • 云服务器
      • 云主机
      • 智能边缘云
      • 应用编排服务
      • 微服务引擎
      • 共享流量包
      更多推荐
      • web应用防火墙
      • 密钥管理
      • 等保咨询
      • 安全专区
      • 应用运维管理
      • 云日志服务
      • 文档数据库服务
      • 云搜索服务
      • 数据湖探索
      • 数据仓库服务
      友情链接
      • 中国电信集团
      • 189邮箱
      • 天翼企业云盘
      • 天翼云盘
      ©2025 天翼云科技有限公司版权所有 增值电信业务经营许可证A2.B1.B2-20090001
      公司地址:北京市东城区青龙胡同甲1号、3号2幢2层205-32室
      • 用户协议
      • 隐私政策
      • 个人信息保护
      • 法律声明
      备案 京公网安备11010802043424号 京ICP备 2021034386号