Java源码深度分析String与StringBuffer及StringBuilder详解

目录
  • StringBuffer和StringBuild的区别
  • 创建StringBuffer()
  • 添加功能
  • 删除功能
  • 替换功能
  • 反转功能
  • 最后总结一下

String的字符串是不可变的,StringBuffer和StringBuilder是可变的

String:是字符常量,适用于少量的字符串操作的情况。

StringBuilder:适用于单线程下在字符缓冲区进行大量操作的情况 。

StringBuffer:适用多线程下在字符缓冲区进行大量操作的情况。

StringBuffer和StringBuild的区别

StingBuffer是线程安全的,StringBuild是线程不安全的。

然后再往上查看,就会发现他两都是调用了AbstractStringBuilder类来实现的。

我们不是说StringBuffer和StringBuilder创建的字符串是可变的。

至于源码就直接看源码,就直接在源码当中详细的体现出来了。

在这篇博客我主要说一下StringBuffer,因为StingBuild和StringBuffer的方法几乎一样,我在这就不做过多的详细赘述了。

创建StringBuffer()

第一种创建方法:

StringBuffer stringBuffer = new StringBuffer();

在使用这个方法创建的Buffer字符串的时候,看底层代码它就调用了这个方法:

    public StringBuffer() {
        super(16);
    }

它虽然没有给传入参数,它是它默认的是一个数组长度为16的字符串。

在往上看这个向上传的父类方法就更加简单明了:

    AbstractStringBuilder(int capacity) {
        value = new char[capacity];
    }

直接创建长度为16的字符型数组。

第二种创建方法:

StringBuffer stringBuffer = new StringBuffer(5);

直接给定要创建的字符长度。

底层源码:

    public StringBuffer(int capacity) {
        super(capacity);
    }

第三种创建方法:

StringBuffer stringBuffer = new StringBuffer(“Xin”);

在创建的时候,初始给定初始化的一个字符串。

那么在这个时候就要考虑它的底层保存字符串的数组要创建几位呢? 看源码,源码会告诉我们一切的。

    public StringBuffer(String str) {
        super(str.length() + 16);
        append(str);
    }

源码当中明确的告诉了我们,在这种情况下创建的字符串,底层数组长度是初始字符串的长度加16.

小小的总结一下:

在创建StringBuffer或StringBuilder字符串的时候如果没有给长度,默认为16的长度,给长度的话,就是给定的长度,给的是字符串的话,默认初始长度就是字符串的长度加16

添加功能

public StringBuffer append(String str)

代码演示一下,这还要一个点考虑一下:

public class Demo01 {
    public static void main(String[] args) {
        StringBuffer stringBuffer = new StringBuffer();
        stringBuffer.append("Xin_");    //stringBuffer.length = 4
        stringBuffer.append("Chen_");   //stringBuffer.length = 9
        stringBuffer.append("Yu_");     //stringBuffer.length = 12
        stringBuffer.append("Chen_");   //stringBuffer.length = 17
        stringBuffer.append("Xi");      //stringBuffer.length = 19
        System.out.println(stringBuffer);
    }
}

在不停的添加下去,当数组的长度到达16的时候,要考虑一下扩容的问题,如何扩容?

老规矩,看源代码,一切都在代码里:

照着这个步骤一步一步的点进去,到达最后的源代码:

MAX_ARRAY_SIZE的长度为:

private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

所以最后是 0x7fffffff-8

总的来说,每次底层数组扩容为上次的2倍加2的长度。

删除功能

第一种删除字符串中指定位置的字符

public StringBuffer deleteCharAt(int index)

这个指定删除字符串当中明确的第几个索引。只删除一个元素

注意这里是索引的位置,不是第几个元素。

public class Demo01 {
    public static void main(String[] args) {
        StringBuffer stringBuffer = new StringBuffer("Xin_Chen_Yu_Chen_Xi");
        System.out.println(stringBuffer);
        stringBuffer.deleteCharAt(4);
        System.out.println(stringBuffer);
    }
}

源码其实也不难,就不重视的看了,在这主要看一经过删除操作,它的字符串长度是如何减少的。

删除字符串当中指定区间的字符串

public StringBuffer delete(int start,int end)

牢记在Java当中大多情况下,数组下标索引都是左闭右开的。

在下面的代码当中,我删除的就是"Chen_"这5个字符

public class Demo01 {
    public static void main(String[] args) {
        StringBuffer stringBuffer = new StringBuffer("Xin_Chen_Yu_Chen_Xi");
        System.out.println(stringBuffer);
        stringBuffer.delete(4,9);
        System.out.println(stringBuffer);
    }
}

和上面一样。看一眼源码,如何处理删除后的字符串的长度:

它两的底层都调用的是native本地反方,交由计算机系统完成。

替换功能

就是将字符串中其中某一段的字符串替换为其他的字符串。

public StringBuffer replace(int start,int end,String str)

public class Demo01 {
    public static void main(String[] args) {
        StringBuffer stringBuffer = new StringBuffer("Xin_Chen_Yu_Chen_Xi");
        System.out.println(stringBuffer);
        stringBuffer.replace(4,8,"Gu_Chen");
        System.out.println(stringBuffer);
    }
}

这里也都是数组的下标索引,所以就是从0开始计数的。

替换这也同上面的一样,也需要考虑一下底层字符串的长度问题,先看源码:

反转功能

这个功能就是将原字符串玩去哪反转过来,倒序保存起来。

public class Demo01 {
    public static void main(String[] args) {
        StringBuffer stringBuffer = new StringBuffer("Xin_Chen_Yu_Chen_Xi");
        System.out.println(stringBuffer);
        stringBuffer.reverse();
        System.out.println(stringBuffer);
    }
}

也来看一下它的源代码

最后总结一下

  • String的字符串是不可变的,StringBuffer和StringBuilder是可变的
  • String:是字符常量,适用于少量的字符串操作的情况。
  • StringBuilder:线程不安全的,适用于单线程下在字符缓冲区进行大量操作的情况 。
  • StringBuffer:线程不安全的,适用多线程下在字符缓冲区进行大量操作的情况。

在这几条总计出来的说法当中,在这再详细说明一下为什么StringBuffer和StringBuilder是可变的。

在文章上面,我展示了许多的源码,仔细观察一下,Buffer和Builder修饰的方法,它最后返回的都是this,返回了,自己,就是无论什么操作,它最终都返回的是自己。

但是有些小伙伴还是有个疑问,当它底层扩容超过它的最大值的时候,它的数组不就要换了吗?

这点确实是对的,但是在这我们要理清一个概念,StringBuffer引用的是底层数组名字的地址,而数组在需要换一个更大的时候,这个时候是数组的引用变化了,就是指向数组的地址变了,但是指向StringBuffer的地址是不变的。

我下来使用一个debug动图来展示一下,相信能更好的理解一下。

主要看控制台里的StringBuffer和value的值。我们发现StringBuffer值地址自始至终都没有改变,而value的值在到达16的时候,需要扩容的时候,就变了。这正印证了我上面所说的。

本篇博客也就到此为止了,感谢你的阅读。如果要是还是想不清楚,特别是最后的总结部分,欢迎在评论区讨论,我看到话,会第一时间共同讨论,解决问题。

这也是我重读javase基础的一个系列,比起当时初学的时候,现在看问题多了个高度,理解什么也相对轻松一点全面一些。学习起来更加偏向阅读源码来看,所以多为大家分享看源码。

到此这篇关于Java源码深度分析String与StringBuffer及StringBuilder详解的文章就介绍到这了,更多相关Java String源码分析内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java String、StringBuilder和StringBuffer的区别详解

    1.共同点: 三者都是字符串类.StringBuilder和StringBuffer都是继承AbstractStringBuilder类,存储字符串都是使用数组进行存储的. 2.区别: a. String 类存储字符串使用的数组被final进行修饰了,当对String对象进行增加修改字符时,都需要重新new一个字符数组,将原来的值拷贝到新的字符数组中,所以当字符串的值需要经常修改的时候就需要考虑用StringBuilder或StringBuffer来进行处理了. public final cla

  • Java 中 String,StringBuffer 和 StringBuilder 的区别及用法

    1 简介 在 Java 语言中,共有 8 个基本的数据类型,分别为:byte.short.int.long.float.double.boolean 和 char,其中 char 类型用于表示单个字符,例如 a.b.c .A.B.C.& 这些大小写字母或者特殊字符等等.在实际的编程中,单个的字符并没有我们想象中用的那么频繁,反而是多个字符组成的"字符串"更为常见,但是在基本的数据类型中,并没有字符串这种数据类型.为了解决这个问题,Java 语言为我们提供了一个被 final 关

  • 详解Java中String,StringBuffer和StringBuilder的使用

    目录 1.String类 2.String对象创建的两种方式 3.String常用方法 4.StringBuffer String和StringBuffer的转换 StringBuffer的常用方法 5.StringBuilder 1.String类 字符串广泛应用 在 Java 编程中,在 Java 中字符串属于对象,Java 提供了 String 类来创建和操作字符串. String对象实现了Serializable接口,说明String对象可以串行化(在网络中进行传输),同时实现了Comp

  • java 中String和StringBuffer与StringBuilder的区别及使用方法

    java 中String和StringBuffer与StringBuilder的区别及使用方法 1. String 类 String的值是不可变的,这就导致每次对String的操作都会生成新的String对象,不仅效率低下,而且大量浪费有限的内存空间. String a = "a"; //假设a指向地址0x0001 a = "b";//重新赋值后a指向地址0x0002,但0x0001地址中保存的"a"依旧存在,但已经不再是a所指向的,a 已经指向

  • java中String、StringBuffer与StringBuilder的区别

    String String类是不可变类,即一旦一个String对象被创建以后,包含在这个对象中的字符序列是不可改变的,直至这个对象被销毁. 这个是String类的解释,之前小咸儿看到这个情况,不能理解上述的解释,如下 String a = "123"; a = "456"; // 打印出来的a为456 System.out.println(a) 看到这里,小咸儿不明白了,这不是明明已经对他进行修改了吗?为什么还说他是一个不可变类呢? 经过小咸儿和小伙伴们的学习,明白

  • Java 中String StringBuilder 与 StringBuffer详解及用法实例

    在Android/Java开发中,用来处理字符串常用的类有3种: String.StringBuilder.StringBuffer. 它们的异同点: 1) 都是 final 类, 都不允许被继承; 2) String 长度是不可变的, StringBuffer.StringBuilder 长度是可变的; 3) StringBuffer 是线程安全的, StringBuilder 不是线程安全的. String VS StringBuffer String 类型和StringBuffer的主要性

  • java中String StringBuffer和StringBuilder的区别详解

    目录 从声明定义上来谈 从结构上来谈 从线程安全来谈 总结 从声明定义上来谈 只有String 可以 直接声明创建 而 StringBuffer 与 StringBuilder 必须去new对象 这是因为只有String会在这种声明方式下去字符串常量池创建,其他则没有 StringBuffer stf = new StringBuffer("abc"); StringBuilder stb = new StringBuilder("abc"); StringBuff

  • Java源码深度分析String与StringBuffer及StringBuilder详解

    目录 StringBuffer和StringBuild的区别 创建StringBuffer() 添加功能 删除功能 替换功能 反转功能 最后总结一下 String的字符串是不可变的,StringBuffer和StringBuilder是可变的 String:是字符常量,适用于少量的字符串操作的情况. StringBuilder:适用于单线程下在字符缓冲区进行大量操作的情况 . StringBuffer:适用多线程下在字符缓冲区进行大量操作的情况. StringBuffer和StringBuild

  • Java源码角度分析HashMap用法

    -HashMap- 优点:超级快速的查询速度,时间复杂度可以达到O(1)的数据结构非HashMap莫属.动态的可变长存储数据(相对于数组而言). 缺点:需要额外计算一次hash值,如果处理不当会占用额外的空间. -HashMap如何使用- 平时我们使用hashmap如下 Map<Integer,String> maps=new HashMap<Integer,String>(); maps.put(1, "a"); maps.put(2, "b&quo

  • Android源码中常用的接口传参实例详解

    Android源码中常用的接口传参实例详解 把MyCclass中的参数传到MyDclass /*接口传参例子2 * MyCclass.java发送MyDclass.java接收 * 原理和MyAclass.java发送MyDclass.java接收完全一样 * */ public class MyCclass { public void getEditext(GetMyFragmentData myFragmentData){ String edStr="人的生命是有限的,可是为人民服务是无限的

  • Go源码字符串规范检查lint工具strchecker使用详解

    目录 1.背景 2.strchecker介绍 3.结论 1.背景 在大型项目开发过程中,经常会遇到打印大量日志,输出信息和在源码中写注释的情况.对于软件开发来说,我们一般都是打印输出英文的日志(主要考虑软件在各种环境下的兼容性,如果打印中文日志可能会出现乱码,另外英文日志更容易搜索,更容易后续做国际化),但是对于我们中国人来说,很容易就把中文全角的中文标点符号一不注意就写到日志中了.不过源码中的注释因为是完全面向开发者的,不会面向客户,所以如果研发团队全是中国人,那么代码注释用中文就更有效率.

  • CentOS 6.6 源码编译安装MySQL 5.7.18教程详解

    一.添加用户和组 1.添加mysql用户组 # groupadd mysql 2.添加mysql用户 # useradd -g mysql -s /bin/nologin mysql -M 二.查看系统中是否安装mysql,如果安装需要卸载 # rpm -qa | grep mysql mysql-libs-5.1.73-3.el6_5.x86_64 # rpm -e mysql-libs-5.1.73-3.el6_5.x86_64 --nodeps 三.安装所需依赖包 # yum -y ins

  • Apache源码安装和虚拟主机配置的教程详解

    源码安装Apache 1.上传Apache源码安装所需软件包 2.安装: 安装顺序 apr->apr-util->pcre->httpd 安装编译环境 yum -y install gcc gcc-c++ apr 编译安装 tar xzf apr-1.4.6.tar.gz cd apr-1.4.6 ./configure --prefix=/usr/local/apr make && make install apr-util 编译安装 tar xzf apr-util-

  • java源码解析之String类的compareTo(String otherString)方法

    一. 前言 最近我发现了一个事情,那就是在面试笔试中,好多公司都喜欢在String字符串上出问题,涉及到方方面面的知识,包括其中的一些常用方法. String 类代表字符串.Java 程序中的所有字符串字面值(如 "abc" )都作为此类的实例实现. 字符串是常量:它们的值在创建之后不能更改.字符串缓冲区支持可变的字符串.因为 String 对象是不可变的,所以可以共享. 近日研究了一下String类的一些方法, 通过查看源码, 对一些常用的方法也有了更透彻的认识, 也让我更加理解了设

  • ASM源码学习之ClassReader、ClassVisitor与ClassWriter详解

    ASM ASM是Java中比较流行的用来读写字节码的类库,用来基于字节码层面对代码进行分析和转换.在读写的过程中可以加入自定义的逻辑以增强或修改原来已编译好的字节码,比如CGLIB用它来实现动态代理.ASM被设计用于在运行时对Java类进行生成和转换,当然也包括离线处理.ASM短小精悍.且速度很快,从而避免在运行时动态生成字节码或转换时对程序速度的影响,又因为它体积小巧,可以在很多内存受限的环境中使用. ASM的主要优势包括如下几个方面: 1. 它又一个很小,但设计良好并且模块化的API,且易于

  • 深入解析Vue源码实例挂载与编译流程实现思路详解

    在正文开始之前,先了解vue基于源码构建的两个版本,一个是 runtime only ,另一个是 runtime加compiler 的版本,两个版本的主要区别在于后者的源码包括了一个编译器. 什么是编译器,百度百科上面的解释是 简单讲,编译器就是将"一种语言(通常为高级语言)"翻译为"另一种语言(通常为低级语言)"的程序.一个现代编译器的主要工作流程:源代码 (source code) → 预处理器 (preprocessor) → 编译器 (compiler) →

  • Laravel源码解析之路由的使用和示例详解

    前言 我的解析文章并非深层次多领域的解析攻略.但是参考着开发文档看此类文章会让你在日常开发中更上一层楼. 废话不多说,我们开始本章的讲解. 入口 Laravel启动后,会先加载服务提供者.中间件等组件,在查找路由之前因为我们使用的是门面,所以先要查到Route的实体类. 注册 第一步当然还是通过服务提供者,因为这是laravel启动的关键,在 RouteServiceProvider 内加载路由文件. protected function mapApiRoutes() { Route::pref

随机推荐