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. OpenHarmony有氧拳击之应用端开发

    一.简介 继<OpenHarmony有氧拳击设备端的开发>后,本次为大家带来酷炫的应用端开发.如下,开发者伴随着音乐,律动出拳后,那开发板屡屡播放"挨打"效果,这究竟是 ...

  2. C# 虚方法virtual详解(转载)

    C# 虚方法virtual详解 在C++.Java等众多OOP语言里都可以看到virtual的身影,而C#作为一个完全面向对象的语言当然也不例外. 虚拟函数从C#的程序编译的角度来看,它和其它一般的函 ...

  3. HarmonyOS:使用MindSpore Lite引擎进行模型推理

      场景介绍 MindSpore Lite是一款AI引擎,它提供了面向不同硬件设备AI模型推理的功能,目前已经在图像分类.目标识别.人脸识别.文字识别等应用中广泛使用. 本文介绍使用MindSpore ...

  4. HDC2021技术分论坛:HarmonyOS本地模拟器重磅来袭!

    作者:longjiangyun,模拟器开发工程师 HarmonyOS模拟器是应用开发者使用IDE进行代码开发.调试.测试等活动中必不可少的工具,它分为本地模拟器和远程模拟器,其中远程模拟器又分为单设备 ...

  5. mysql 必知必会整理—sql 正则表达[五]

    前言 简单整理一下sql 正则表达式. 正文 正则表达式是用来匹配文本的特殊的串(字符集合).如果你想从一个文本文件中提取电话号码,可以使用正则表达式.如果你需要查找名字中间有数字的所有文件,可以使用 ...

  6. Pytorch-实战之对Himmelblau函数的优化

    1.Himmelblau函数 Himmelblau函数: F(x,y)=(x²+y-11)²+(x+y²-7)²:具体优化的是,寻找一个最合适的坐标(x,y)使得F(x,y)的值最小. 函数的具体图像 ...

  7. JS中通过url动态获取图片大小的方法小结(两种方法)

    很多时候再项目中,我们往往需要先获取图片的大小再加载图片,但是某些特定场景,如用过cocos2d-js的人都知道,在它那里只能按比例缩放大小,是无法设置指定大小的图片的,这就是cocos2d-js 的 ...

  8. LLM开源小工具(基于代码库快速学习/纯shell调用LLM灵活管理系统)

    随着AI的各种信息的发展,LLM各种模型不断涌现,作为一名IT人员不得不向前走,不断探索学习发现新知识. 随着学习,也了解到一些对于模型的调用,从而解决一些问题,或者对已有工具或应用的重写.如下是两个 ...

  9. Oracle常用的创建表语句

    Oracle常用的创建表语句 Oracle常用的创建表语句 指定字段的创建 --指定字段的创建 create table table_name( test_1(字段名1) varchar2(50),( ...

  10. 力扣608(MySQL)-树节点(中等)

    题目: 给定一个表 tree,id 是树节点的编号, p_id 是它父节点的 id . 树中每个节点属于以下三种类型之一: 叶子:如果这个节点没有任何孩子节点.根:如果这个节点是整棵树的根,即没有父节 ...