深入底层谈谈String

news2025/1/20 5:55:14

深入底层谈谈String

  • 一、聊聊字符串拼接【底层】
  • 二、聊聊String实现(源码分析)
    • 实现的接口
    • 内部属性及其部分构造函数
    • 部分方法说明
      • 明明replace,replaceAll,substring等方法得到了新的字符串,为什么说String是不变的呢?
  • 三、总结

一、聊聊字符串拼接【底层】

给一个测试 Demo1:

public class Demo1 {
    public static void main(String[] args) {
        String a = "a";
        String b = "b";
        String c = "ab";
        String d = a+b;
    }
}

利用 javap -v Demo1.class 命令看看编译后字节码中的main方法(
-v 命令参数,含附加信息)

public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: (0x0009) ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=5, args_size=1
         0: ldc           #2                  // String a
         2: astore_1    放到下面局部变量表中,1表示的是位置,对应Slot列名
         3: ldc           #3                  // String b
         5: astore_2
         6: ldc           #4                  // String ab
         8: astore_3
         9: new           #5      这里准备实例化了一个StringBuilder对象            // class java/lang/StringBuilder
        12: dup
        13: invokespecial #6       通过无参构造的方式           // Method java/lang/StringBuilder."<init>":()V
        16: aload_1		从表中取变量,取Slot为1的
        17: invokevirtual #7    这里对拿到的变量进行字符串凭借             // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        20: aload_2
        21: invokevirtual #7       这里也对拿到的变量进行字符串凭借           // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        24: invokevirtual #8 这里通过调用StringBuilder中的toString方法得到一个新的String对象            // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
        27: astore        4     把那得到的新String对象赋给局部变量表中Slot为4的
        29: return   方法结束
      // 这是main方法中的局部变量表  
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      30     0  args   [Ljava/lang/String;
            3      27     1     a   Ljava/lang/String;
            6      24     2     b   Ljava/lang/String;
            9      21     3     c   Ljava/lang/String;
           29       1     4     d   Ljava/lang/String;

通过分析上面(含附加信息)的main方法的字节码可以得出,字符串对象通过+进行拼接的细节部分。首先会去创建一个空的StringBuilder对象,然后把从局部变量表中取出相应的字符串调用append方法进行拼接,最后通过toString方法得出结果也就是说每次使用+去拼接字符串都会创建一个新的StringBuilder对象去进行操作。所以如果是什么什么集合字符串然后进行拼接,还是别乱用加号进行拼接。

StringBuilder中的toString方法
    @Override
    public String toString() {
        // Create a copy, don't share the array
        return new String(value, 0, count);
    }

可以看见是重新 new 了一个 String 对象,也就是 d 变量是指向一个新的String 对象,放入堆中。 但是上面的 c 变量是通过直接字符串赋值的方式,这种方式是先会在字符串常量池中查找,如果存在,则返回该字符串的引用,否则就在字符串常量池中创建一个新的字符串对象,并返回该字符串对象的引用。 后者是内存中就一份,可以被多个引用共享;前者是怎么都会在堆中产生对应的新空间给其使用。

下面对 c 和 d 对象地址进行比对会输出 false。

System.out.println(c==d);// false

所以

建议在使用字符串时,尽量使用直接字符串赋值方式,以便复用字符串常量池中的对象,避免不必要的内存占用和性能问题。

再来看看直接通过字符串的方式进行拼接。

	String e = "a" + "b";

字节码分析

在这里插入图片描述也就是和对象 c 指向应该为同一个地址。

System.out.println(c==e);//true

二、聊聊String实现(源码分析)

实现的接口

在这里插入图片描述public final class String implements java.io.Serializable, Comparable<String>, CharSequence

从这可以得出String类是final类型的,不允许你瞎拓展。
还有就是实现了Comparable接口,可以通过compareTo重写的方法进行字符串比较。

内部属性及其部分构造函数

内部属性

在这里插入图片描述重点看那个value数组,它是final类型的,这也是为什么说String字符串定义后永远不会变的原因。

构造函数,重点有几个

在这里插入图片描述这就是直接拿参数的value进行拷贝。

public String(String original) {
        this.value = original.value;
        this.hash = original.hash;
    }

这个就是调用copyOf方法==》System.arraycopy方法去进行拷贝。

public String(char value[]) {
        this.value = Arrays.copyOf(value, value.length);
    }

部分方法说明

  • equals方法(小编还是觉得挺简单的,没什么好说的,需要注意的是,虽然value是私有的,但传进来的String对象是在本类中使用的,所以也是可以调用value属性的。)
    public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        if (anObject instanceof String) {
            String anotherString = (String)anObject;
            int n = value.length;
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                        return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }

明明replace,replaceAll,substring等方法得到了新的字符串,为什么说String是不变的呢?

为什么String对象是不可变的?

其实那个final类型的value数组就很明显了,至于方法为什么返回的是新的字符串,看源码的话非常清晰。这里拿一个replace和substring方法说明一下。

在这里插入图片描述

在这里插入图片描述
可以看见都是去new一个新的String对象。

三、总结

  • 字符串对象之间的拼接底层会创建一个空的StringBuilder对象,然后通过append方法进行拼接;如果是纯字符串拼接,那么底层会直接进行拼接,然后去字符串常量池中找。(字符串常量池是通过哈希表实现的,如果字符串存在的话就会返回其对应的引用)
  • String对象不可变性、不可拓展性源于起内部实现是一个final类型的value数组,String类是final型的。
  • substring等方法是返回一个新的String对象。

当然上面说的是jdk8中的String对象,在jdk9后String对象做了很多改变,比如底层不再是char数组,而是byte数组,高效的使用内存;还新增了一些静态方法…

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/418447.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

kali设置静态ip地址

Kali设置静态ip地址 本篇文章主要分3部分讲解如何为Kali虚拟机配置静态IP地址&#xff0c;使其能够与主机和外网进行通信。首先需要配置VMware的虚拟网卡&#xff0c;然后配置Kali虚拟机&#xff0c;最后配置主机。 一、配置VMware 打开VMware&#xff0c;选择【编辑】—【虚…

C++游戏分析与破解方法介绍

1、C游戏简介 目前手机游戏直接用C开发的已经不多&#xff0c;使用C开发的多是早期的基于cocos2dx的游戏&#xff0c;因此我们这里就以cocos2d-x为例讲解C游戏的分析与破解方法。 Cocos2d-x是一个移动端游戏开发框架&#xff0c;可以使用C或者lua进行开发&#xff0c;也可以混…

Spring框架核心与设计思想

文章目录一、Spring是什么&#xff1f;二、什么是IoC容器&#xff1f;什么是IOC&#xff1f;Spring IoC三、DI总结一、Spring是什么&#xff1f; 我们一般所说的Spring指的是Spring Framework(Spring 框架)&#xff0c;它是一个开源的框架&#xff0c;Spring支持广泛的应用场景…

Spring事务详解

&#x1f3c6;今日学习目标&#xff1a; &#x1f340;Spring事务详解 ✅创作者&#xff1a;林在闪闪发光 ⏰预计时间&#xff1a;30分钟 &#x1f389;个人主页&#xff1a;林在闪闪发光的个人主页 &#x1f341;林在闪闪发光的个人社区&#xff0c;欢迎你的加入: 林在闪闪发光…

贯穿设计模式第三话--依赖倒转原则

&#x1f973;&#x1f973;&#x1f973; 茫茫人海千千万万&#xff0c;感谢这一刻你看到了我的文章&#xff0c;感谢观赏&#xff0c;大家好呀&#xff0c;我是最爱吃鱼罐头&#xff0c;大家可以叫鱼罐头呦~&#x1f973;&#x1f973;&#x1f973; 从今天开始&#xff0c;将…

【GPT4】GPT4 官方报告解读

欢迎关注【youcans的AGI学习笔记】原创作品 【GPT4】GPT-4 官方报告解读1. GPT-4 官方介绍2. GPT-4 的性能2.1 GPT-4 在各种学术和专业考试中的性能2.2 GPT-4 在传统机器学习测试中的性能2.3 GPT-4 在不同语言测试中的性能3. GPT-4 的图像输入功能3.1 GPT-4 图像输入案例3.2 GP…

《分解因数》:质因数分解

目录 一、题目&#xff1a; 二、思路&#xff1a; 三、代码&#xff1a; 一、题目&#xff1a; 分解因数 《分解因数》题目链接 所谓因子分解&#xff0c;就是把给定的正整数a&#xff0c;分解成若干个素数的乘积&#xff0c;即 a a1 a2 a3 ... an,并且 1 < a1…

<数据结构> 链表 - 单链表(c语言实现)

B.最简单结构的链表——不带哨兵位单链表的实现&#xff08;关于哨兵位结点&#xff09; 一、不带哨兵位单链表结点的创建1.1 typedef 链表的数据类型 1.2 结点的结构体创建 二、单链表要实现的功能 三、需要包含的头文件四、函数接口一览为什么有些函数参数传递的是二级指针&a…

【FreeRTOS(二)】FreeRTOS新手入门——计数型信号量和二进制信号量的基本使用并附代码解析

写在前面&#xff1a; 本文章如有错漏之处&#xff0c;敬请指正&#xff0c;另外本文为网络材料整理&#xff0c;侵删。 FreeRTOS信号量的基本使用&代码解析一、信号量概述二、计数型信号量三、二进制信号量四、信号量函数API1、创建信号量2、删除一个信号量3、信号量释放4…

ASP.NET动态Web开发技术第5章

第5章数据验证一.预习笔记 1.验证控件概述&#xff1a; 2.RequiredFieldValidator&#xff08;必填验证&#xff09; 常用属性1&#xff1a;ControlToValidator:被验证的输入控件的ID 常用属性2&#xff1a;Text&#xff1a;验证失败时&#xff0c;验证控件显示的文本 常用…

8.3 总体分布的假设检验

学习目标&#xff1a; 如果我要学习总体分布的假设检验&#xff0c;我会采取以下步骤&#xff1a; 掌握基础概念&#xff1a;学习和掌握统计学中基础的概念&#xff0c;如总体、样本、假设检验、p值等等。 学习检验方法&#xff1a;了解和学习不同的总体分布假设检验方法&…

亚信科技AntDB数据库荣膺第十二届数据技术嘉年华(DTC 2023)“最具潜力数据库”大奖

近日&#xff0c;亚信科技AntDB数据库产品在第十二届数据技术嘉年华&#xff08;DTC 2023&#xff09;峰会上斩获“2022年度最具潜力数据库”大奖。亚信安慧副总裁张桦先生受邀参会&#xff0c;并发表了《AntDB数据库通信行业核心系统应用与创新》的主题演讲&#xff0c;分享了…

vue实现好看的相册、图片网站

目录 一、效果图 1.项目访问地址 2.画虫官方效果图&#xff1a; 3.作者实现的效果图&#xff1a; 二、代码实现 1.项目结构截图 2.路由配置代码&#xff1a; 3. 头部底部主页面内容显示容器的代码 4.首页&#xff0c;即标签页的代码 三、项目启动说明 四、总结 一、…

Android---MVC/MVP/MVVM的演进

目录 一个文件打天下 一个文件--->MVC MVC--->MVP MVP--->MVVM 6大设计原则 完整demo 我们通过"#字棋"游戏来展现MVC-->MVP-->MVVM 之间的演进 一个文件打天下 数据、视图以及逻辑都放在一个 class 里面。而一个 class 里最多 500 行代码&…

springboot 密码加密

首先介绍一下jasypt的使用方法 版本对应的坑 使用的时候还是遇到一个坑&#xff0c;就是jasypt的版本与spring boot版本存在对应情况。可以看到jasypt是区分java7和java8的&#xff0c;也存在依赖spring版本的情况。 自己尝试了一下 在使用jasypt-spring-boot-starter的前提…

优思学院|职场达人有什么晋升秘诀?

作为职场人士&#xff0c;升职晋升是我们一直追求的目标。然而&#xff0c;在职场中&#xff0c;竞争是激烈的&#xff0c;只有那些真正做到了突出表现和积极进取的人才能获得晋升机会。这里将分享七个职场达人的晋升秘诀&#xff0c;希望对那些正在寻找升职机会的人有所帮助。…

Python圈的普罗米修斯——一套近乎完善的监控系统

文章目录前言一、怎么采集监控数据&#xff1f;二、采集的数据结构与指标类型2.1 数据结构2.2 指标类型2.3 实例概念2.4.数据可视化2.5.应用前景总结前言 普罗米修斯(Prometheus)是一个SoundCloud公司开源的监控系统。当年&#xff0c;由于SoundCloud公司生产了太多的服务&…

SQL综合查询下

SQL综合查询下 目录SQL综合查询下18、查询所有人都选修了的课程号与课程名题目代码题解19、SQL查询&#xff1a;查询没有参加选课的学生。题目代码20、SQL查询&#xff1a;统计各门课程选修人数&#xff0c;要求输出课程代号&#xff0c;课程名&#xff0c;有成绩人数&#xff…

Express使用

文章目录Express 使用概述下载Express简单使用Express 生成器安装生成器使用基本路由使用路由获取请求数据获取路由参数处理请求体设置响应方式一&#xff1a;兼容http模块方式二&#xff1a;express的响应方法其他响应中间件简介全局中间件路由中间件静态资源中间件Router简介…

SkyWalking服务应用

文章目录SkyWalking服务应用案例准备案例实施1.部署Elasticsearch服务2.部署SkyWalking OAP服务3.部署SkyWalking UI服务4.搭建并启动应用商城服务SkyWalking服务应用 案例准备 节点规划 IP主机名节点192.168.100.10node-1Skywalking实验节点192.168.100.20mall商城搭建节点…