StringBuilder结构

  • 明显的看到StringBuilder跟String相似的存在char数组
  • 区别是StringBuilder的char[]数组不是final修饰的,所以是可以多次改变的,随时可以改变源码
1 abstract class AbstractStringBuilder implements Appendable, CharSequence {
2 /**
3 * The value is used for character storage.
4 */
5 char[] value;
6
7 ....

核心要点

  • final的有无区别
  • 可以成为可变的字符序列

StringBuffer结构

  • 明显的看到StringBuffer跟String相似的存在char数组
  • 区别是StringBuffer的char[]数组不是final修饰的,所以是可以多次改变的,随时可以改变源码
    1 abstract class AbstractStringBuilder implements Appendable, CharSequence {
    2 /**
    3 * The value is used for character storage.
    4 */
    5 char[] value;
    6
    7 ....
    8 }

核心要点

  • final的有无区别
  • 可以成为可变的字符序列

StringBuffer和StringBuilder区别

不同:

  • StringBuffer线程安全,效率低
  • StringBuilder线程不安全,效率高
  • 一般使用StringBuilder,因为一般不涉及线程安全

相同:

  • 都是AbstractStringBuilder子类

实例:

可变字符序列,内容可以随意修改

 1       StringBuilder stringBuilder = new StringBuilder("abcdfg");
2
3 System.out.println(Integer.toHexString(stringBuilder.hashCode()));
4
5 System.out.println(stringBuilder);
6
7
8 stringBuilder.setCharAt(3, 'L');
9 System.out.println(Integer.toHexString(stringBuilder.hashCode()));
10
11 System.out.println(stringBuilder);

基本用法

    • 添加26个字母

       1        // 初始化StringBuffer
      2 StringBuffer twentySixLetter = new StringBuffer();
      3
      4 for (int i = 0; i < 26; i++) {
      5
      6 // 字符增加
      7 char needTransformationLetter = (char) ('a' + i);
      8
      9 // 添加字母到stringBuffer
      10 twentySixLetter.append(needTransformationLetter);
      11
      12 }
      13
      14 System.out.println("打印26个字母:" + twentySixLetter);
    • 倒叙字符序列
      1         twentySixLetter.reverse();
      2
      3 System.out.println("打印26个字母的倒叙:" + twentySixLetter);
    • 修改制定位置的字符
      1         twentySixLetter.setCharAt(6,'走');
      2 System.out.println("修改第六个位置的字符:" + twentySixLetter);
    • 指定位置插入字符
      1         twentySixLetter.insert(0,'再');
      2 System.out.println("在第一个位置插入字符:" + twentySixLetter);

注:

观察源码

// 继承父类方法

super.insert(offset, c);

// 返回本身对象

return this;

这就意味着可以多次的使用这个对象,一般称之为 链式调用

核心就是:

调用了 return this;

如下:

1        // 指定位置插入字符
2 twentySixLetter.insert(0,'再').insert(1,'三');
3 System.out.println("在第一个位置插入字符:" + twentySixLetter);
  • 删除某个区间的字符,或删除某个位置的字符 delete也是链式调用,也可以连续删除

    1        // 删除区间字符
    2 twentySixLetter.delete(20,26);
    3 System.out.println("删除后的字符:" + twentySixLetter);
  • 删除某个字符
    1         //删除某个字符
    2 twentySixLetter.deleteCharAt(0).deleteCharAt(0);
    3 System.out.println("删除第一个第二个字符:" + twentySixLetter);
  • 获取某个字符
    1         //获取字符
    2 twentySixLetter.charAt(15);
    3 System.out.println("获取字符:" + twentySixLetter);

不可变和可变字符序列使用陷阱

  • 循环累加字符串的时候,不建议如下写法,会创建大量对象,占用过多资源,消耗服务器资源

    1         String wrongWriteString = "";
    2
    3 for (int i = 0; i < 5000; i++) {
    4
    5 //相当于产生了10000个对象
    6 wrongWriteString = wrongWriteString + i;
    7 }
    8
    9 System.out.println(wrongWriteString);
  • 建议如下写法,可以减少创建对象和所消耗的时间
    1        StringBuilder stringBuilderRight = new StringBuilder("");
    2
    3 for (int i = 0; i < 5000; i++) {
    4 stringBuilderRight.append(i);
    5 }
    6
    7 System.out.println(stringBuilderRight);
  • 效率,占用内存测试
     1         // 使用String进行字符串的拼接
    2 String stringWrong = "";
    3 //本质上使用StringBuilder拼接, 但是每次循环都会生成一个StringBuilder对象
    4 //获取系统剩余内存空间
    5 long num1 = Runtime.getRuntime().freeMemory();
    6 //获取系统的当前时间
    7 long time1 = System.currentTimeMillis();
    8 for (int i = 0; i < 5000; i++) {
    9 //相当于产生了10000个对象
    10 stringWrong = stringWrong + i;
    11 }
    12 long num2 = Runtime.getRuntime().freeMemory();
    13 long time2 = System.currentTimeMillis();
    14 System.out.println("String占用内存 : " + (num1 - num2));
    15 System.out.println("String占用时间 : " + (time2 - time1));
    16
    17
    18 /**使用StringBuilder进行字符串的拼接*/
    19 StringBuilder stringBuilderRight = new StringBuilder("");
    20 long num3 = Runtime.getRuntime().freeMemory();
    21 long time3 = System.currentTimeMillis();
    22 for (int i = 0; i < 5000; i++) {
    23 stringBuilderRight.append(i);
    24 }
    25 long num4 = Runtime.getRuntime().freeMemory();
    26 long time4 = System.currentTimeMillis();
    27 System.out.println("StringBuilder占用内存 : " + (num3 - num4));
    28 System.out.println("StringBuilder占用时间 : " + (time4 - time3));

JDK源码阅读-------自学笔记(十二)(java.lang.StringBuffer和StringBuilder比较)的更多相关文章

  1. JDK源码阅读-------自学笔记(一)(java.lang.Object重写toString源码)

    一.前景提要 Object类中定义有public String toString()方法,其返回值是 String 类型. 二.默认返回组成 类名+@+16进制的hashcode,当使用打印方法打印的 ...

  2. JDK源码阅读-------自学笔记(二十五)(java.util.Vector 自定义讲解)

    Vector 向量 Vector简述 1).Vector底层是用数组实现的List 2).虽然线程安全,但是效率低,所以并不是安全就是好的 3).底层大量方法添加synchronized同步标记,sy ...

  3. JDK源码阅读-------自学笔记(五)(浅析数组)

    一.数组基础 1.定义和特点 数组也可以看做是对象,数组变量属于引用类型,数组中每个元素相当于该队形的成员变量,数组对象存储在堆中. 2.初始化数组 常用类初始化 // 整型初始化 int[] int ...

  4. JDK源码阅读-------自学笔记(二十四)(java.util.LinkedList 再探 自定义讲解)

    一.实现get方法 1.一般思维实现思路 1).将对象的值放入一个中间变量中. 2).遍历索引值,将中间量的下一个元素赋值给中间量. 3).返回中间量中的元素值. 4).示意图 get(2),传入角标 ...

  5. 利用IDEA搭建JDK源码阅读环境

    利用IDEA搭建JDK源码阅读环境 首先新建一个java基础项目 基础目录 source 源码 test 测试源码和入口 准备JDK源码 下图框起来的路径就是jdk的储存位置 打开jdk目录,找到sr ...

  6. JDK源码阅读-ByteBuffer

    本文转载自JDK源码阅读-ByteBuffer 导语 Buffer是Java NIO中对于缓冲区的封装.在Java BIO中,所有的读写API,都是直接使用byte数组作为缓冲区的,简单直接.但是在J ...

  7. JDK源码阅读(三):ArraryList源码解析

    今天来看一下ArrayList的源码 目录 介绍 继承结构 属性 构造方法 add方法 remove方法 修改方法 获取元素 size()方法 isEmpty方法 clear方法 循环数组 1.介绍 ...

  8. JDK源码阅读(一):Object源码分析

    最近经过某大佬的建议准备阅读一下JDK的源码来提升一下自己 所以开始写JDK源码分析的文章 阅读JDK版本为1.8 目录 Object结构图 构造器 equals 方法 getClass 方法 has ...

  9. 手机自动化测试:appium源码分析之bootstrap十二

    手机自动化测试:appium源码分析之bootstrap十二   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣 ...

  10. JDK源码阅读(1)_简介+ java.io

    1.简介 针对这一个版块,主要做一个java8的源码阅读笔记.会对一些在javaWeb中应用比较广泛的java包进行精读,附上注释.对于容易混淆的知识点给出相应的对比分析. 精读的源码顺序主要如下: ...

随机推荐

  1. #树状数组#洛谷 5677 [GZOI2017]配对统计

    题目 分析 考虑处理出所有右端点的能够匹配的左端点,然后用树状数组离线查询 代码 #include <cstdio> #include <cctype> #include &l ...

  2. Apache Maven ToolChains的使用

    目录 简介 Toolchains的介绍 Toolchains的例子 Toolchains支持 总结 简介 Maven是java中非常有用和常用的构建工具,基本上现在大型的java项目都是Maven和g ...

  3. 熊磊:成功移植 OpenHarmony 到多套开发板,是最开心的事

    编者按:在 OpenHarmony 生态发展过程中,涌现了大批优秀的代码贡献者,本专题旨在表彰贡献.分享经验,文中内容来自嘉宾访谈,不代表 OpenHarmony 工作委员会观点. 熊磊 华为技术有限 ...

  4. 1开幕在即 | “万物互联,使能千行百业”2022开放原子全球开源峰会OpenAtom OpenHarmony分论坛

    7月27日下午,聚焦开源产业与生态的2022开放原子全球开源峰会OpenAtom OpenHarmony分论坛将在北京亦创国际会展中心盛大开幕. 作为OpenHarmony工作委员会联合生态合作伙伴为 ...

  5. Java 内存分析(程序实例),学会分析内存,走遍天下都不怕!!!

    相信大多数的java初学者都会有这种经历:碰到一段代码的时候,不知该从何下手分析,不知道这段代码到底是怎么运行最后得到结果的..... 等等吧,很多让人头疼的问题,作为一名合格的程序员呢,遇到问题一定 ...

  6. Taurus.MVC 性能压力测试(ap 压测 和 linux 下wrk 压测):.NET Core 版本

    前言: 最近的 Taurus.MVC 版本,对性能这一块有了不少优化,因此准备进行一下压测,来测试并记录一下 Taurus.MVC 框架的性能,以便后续持续优化改进. 今天先压测 .NET Core  ...

  7. Python3中pip3命令的用法介绍及安装配置

    第一节:pip3是什么?有啥用? pip3:(Python3 Install Package ),这个英文全称是我为了更好的理解这个命令这么叫的,官方没有这对个命令的全称的解释:) python 支持 ...

  8. 【未测试】CentOS 6.5快速部署HTTP WEB服务器和FTP服务器

    CentOS 6.5快速部署HTTP WEB服务器和FTP服务器 [题记]本文使用CentOS 6.5minimal快速搭建HTTP服务器和仅供授权用户登陆的FTP服务器.意在使用授权FTP用户通过登 ...

  9. mysql 必知必会整理—组合查询与全文搜索[九]

    前言 简单整理一下组合查询与全文搜索. 正文 什么是组合查询,就是我们常说的交并补集. 直接上例子. 举一个例子,假如需要价格小于等于5的所有物品的一个列表,而且还想包括供应商1001和1002生产的 ...

  10. 论文记载: Deep Reinforcement Learning for Traffic LightControl in Vehicular Networks

    强化学习论文记载 论文名: Deep Reinforcement Learning for Traffic LightControl in Vehicular Networks ( 车辆网络交通信号灯 ...