从Java的Jvm的角度解释一下为什么String不可变?

news2025/6/1 23:42:50

从Java的Jvm的角度解释一下为什么String不可变?

从 JVM 的角度看,Java 中 String 的不可变性是由多层次的机制共同保障的,这些设计涉及内存管理、性能优化和安全保障:


1. JVM 内存模型与字符串常量池

  • 字符串常量池(String Pool)
    JVM 在堆内存中维护一个特殊的区域(Java 7 前在永久代,Java 7+ 在堆中)。当创建字符串字面量时:

    String s1 = "hello";  // 首次创建,在常量池分配内存
    String s2 = "hello";  // 复用常量池中的"hello"
    
    • s1s2 指向同一内存地址s1 == s2true)。
    • 如果 String 可变:修改 s1 会导致 s2 的值意外改变,破坏程序逻辑。
  • 技术实现
    JVM 通过 intern() 方法实现常量池机制。编译时确定的字面量自动入池,运行时可通过 intern() 手动入池。


2. 对象存储结构的不可变性

Java 8 及以前
public final class String {
    private final char value[]; // final 修饰的字符数组
    private final int hash;     // 缓存哈希值
}
  • final 关键字的作用
    1. value 引用不可变(不能指向新数组)
    2. 数组内容虽可通过反射修改,但破坏封装性(非正常操作)
Java 9+ 的优化
private final byte[] value;     // 改为字节数组(节省内存)
private final byte coder;       // 编码标记 (LATIN1/UTF16)

即使底层存储优化,数组引用和内容仍不可变


3. JVM 安全机制

  • 类加载安全
    字符串用于类全限定名(如 java.lang.Object)。如果字符串可变:

    • 恶意代码可修改类名字符串,破坏 JVM 类加载机制。
    • 导致类型系统混乱(如篡改 "java.lang.Integer" 为恶意类名)。
  • 访问控制安全
    字符串用于文件路径、网络地址等敏感信息:

    String path = "/etc/passwd";
    File file = new File(path);
    

    如果 path 在传递后被修改,可能导致安全漏洞。


4. 性能优化

  • 哈希码缓存
    String 重写了 hashCode(),首次计算后缓存结果:

    public int hashCode() {
        int h = hash;  // 默认为0
        if (h == 0 && value.length > 0) {
            // 计算哈希值并缓存
            hash = h = ...; 
        }
        return h;
    }
    

    依赖不可变性:如果内容可变,缓存的哈希值会失效。

  • 同步与线程安全
    不可变对象天生线程安全,无需同步:

    // 多线程共享安全
    public static final String GLOBAL_CONFIG = "config";
    

    String 可变,全局常量将需要同步锁。


5. 编译器与运行时的协作

  • 编译期优化
    编译器合并已知字面量:

    String s = "a" + "b" + "c"; // 编译为 String s = "abc";
    

    依赖字符串不可变才能安全优化。

  • 运行时优化
    JVM 内联字符串操作,基于不可变性假设生成高效机器码。


6. 不可变性的"漏洞"与防御

虽然理论上可通过反射修改数组内容:

String str = "hello";
Field field = String.class.getDeclaredField("value");
field.setAccessible(true);
char[] value = (char[]) field.get(str);
value[0] = 'H'; // 修改为"Hello"

但 JVM 通过以下方式防御:

  1. SecurityManager 可阻止此类操作
  2. 破坏常量池一致性(修改后所有引用 "hello" 的位置均受影响)
  3. 实际开发中禁止使用(违反语言设计原则)

为什么 JVM 要这样设计?

  1. 内存效率
    常量池减少重复字符串内存占用(相同内容共享一份)。

  2. 性能优势

    • 哈希码缓存加速 HashMap 等集合操作
    • 避免同步开销
    • 编译器/运行时深度优化
  3. 系统安全
    保障类加载、访问控制等核心机制安全。

  4. 设计一致性
    作为基础数据类型,不可变性简化程序逻辑(如 Map<String, ...> 的键安全)。


不可变性的代价与解决方案

  • 拼接性能问题
    频繁修改字符串时产生临时对象:
    String s = "";
    for (int i = 0; i < 1000; i++) {
        s += i; // 产生大量临时 String 和 StringBuilder
    }
    
  • 解决方案
    使用 StringBuilder(单线程)或 StringBuffer(多线程)替代。

总结:JVM 视角下的不可变性

层面机制
内存模型字符串常量池依赖不可变性实现对象复用
对象结构final 字段 + 私有数组保障存储不可变
安全机制防止类加载、访问控制等核心功能被篡改
性能优化哈希码缓存、线程安全、编译器优化
JVM 规范Java 语言规范强制要求(§4.3.3. Strings are constant)

正是 JVM 从内存管理到运行时优化的全方位设计,确保了 String 不可变性的严格执行,成为 Java 体系稳定性的基石。

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

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

相关文章

历年西安交通大学计算机保研上机真题

2025西安交通大学计算机保研上机真题 2024西安交通大学计算机保研上机真题 2023西安交通大学计算机保研上机真题 在线测评链接&#xff1a;https://pgcode.cn/school 计算圆周率近似值 题目描述 根据公式 π / 4 1 − 1 / 3 1 / 5 − 1 / 7 … \pi / 4 1 - 1/3 1/5 - …

基本数据指针的解读-C++

1、引言 笔者认为对于学习指针要弄清楚如下问题基本可以应付大部分的场景&#xff1a; ① 指针是什么&#xff1f; ② 指针的类型是什么&#xff1f; ③ 指针指向的类型是什么&#xff1f; ④ 指针指向了哪里&#xff1f; 2、如何使用指针 使用时的步骤如下&#xff1a; ① …

基于多模态脑电、音频与视觉信号的情感识别算法【Nature核心期刊,EAV:EEG-音频-视频数据集】

简述 理解情感状态对于开发下一代人机交互界面至关重要。社交互动中的人类行为会引发受感知输入影响的心理生理过程。因此&#xff0c;探索大脑功能与人类行为的努力或将推动具有类人特质人工智能模型的发展。这里原作者推出一个多模态情感数据集&#xff0c;包含42名参与者的3…

【QueryServer】dbeaver使用phoenix连接Hbase(轻客户端方式)

一、轻客户端连接方式 (推荐) 演示无认证配置方式, 有认证填入下方有认证参数即可 1, 新建连接 → Hadoop/大数据 → Apache Phoenix 2, 手动配置QueryServer驱动: 填入: “类名”, “URL模版”(注意区分有无认证), “端口号”, (勾选无认证) 类名: org.apache.phoenix…

[9-1] USART串口协议 江协科技学习笔记(13个知识点)

1 2 3 4全双工就是两个数据线&#xff0c;半双工就是一个数据线 5 6 7 8 9 10 TTL&#xff08;Transistor-Transistor Logic&#xff09;电平是一种数字电路中常用的电平标准&#xff0c;它使用晶体管来表示逻辑状态。TTL电平通常指的是5V逻辑电平&#xff0c;其中&#xff1a;…

Oracle基础知识(五)——ROWID ROWNUM

目录 一、ROWID 伪列 二、ROWNUM——限制查询结果集行数 1.ROWNUM使用介绍 2.使用ROWNUM进行分页查询 3.使用ROWNUM查看薪资前五位的员工 4.查询指定条数直接的数据 三、ROWNUM与ROWID不同 一、ROWID 伪列 表中的每一行在数据文件中都有一个物理地址&#xff0c;ROWID…

EMS只是快递那个EMS吗?它跟能源有什么关系?

在刚刚落幕的深圳人工智能终端展上&#xff0c;不少企业展示了与数字能源相关的技术和服务&#xff0c;其中一项关键系统——EMS&#xff08;Energy Management System&#xff0c;能量管理系统&#xff09;频频亮相。这个看似低调的名字&#xff0c;实际上正悄然成为未来能源管…

日志技术-LogBack、Logback快速入门、Logback配置文件、Logback日志级别

一. 日志技术 1. 程序中的日志&#xff0c;是用来记录应用程序的运行信息、状态信息、错误信息等。 2. JUL&#xff1a;(java.util.logging)这是JavaSE平台提供的官方日志框架&#xff0c;也被称为JUL。配置相对简单&#xff0c;但不够灵活&#xff0c;性能较差。 3.Logs4j&…

修改Cinnamon主题

~/.themes/Brunnera-Dark/cinnamon/cinnamon.css 1.修改 Tooltip 圆角大小&#xff0c;边框颜色&#xff0c;背景透明度 #Tooltip { border-radius: 10px; color: rgba(255, 255, 255, 0.8); border: 1px solid rgba(255, 255, 255, 0.6); background-color: rgba(0,…

91.评论日记

2025年5月30日20:27:06 AI画减速器图纸&#xff1f; 呜呜为什么读到机械博士毕业了才有啊 | 新迪数字2025新品发布会 | AI工业软件 | 三维CAD | 国产自主_哔哩哔哩_bilibili

HTML5实现简洁的端午节节日网站源码

HTML5实现简洁的端午节节日网站源码 前言一、设计来源1.1 网站首页界面1.2 端午由来界面1.3 节日活动界面1.4 传统美食界面1.5 民俗文化界面1.6 登录界面1.7 注册界面 二、效果和源码2.1 动态效果2.2 源代码 结束语 HTML5实现简洁的端午节节日网站源码&#xff0c;酷炫的大气简…

Window10+ 安装 go环境

一、 下载 golang 源码&#xff1a; 去官网下载&#xff1a; https://go.dev/dl/ &#xff0c;当前时间&#xff08;2025-05&#xff09;最新版本如下: 二、 首先在指定的磁盘下创建几个文件夹 比如在 E盘创建 software 文件夹 E:\SoftWare,然后在创建如下几个文件夹 E:\S…

一、Sqoop历史发展及原理

作者&#xff1a;IvanCodes 日期&#xff1a;2025年5月30日 专栏&#xff1a;Sqoop教程 在大数据时代&#xff0c;数据往往分散存储在各种不同类型的系统中。其中&#xff0c;传统的关系型数据库 (RDBMS) 如 MySQL, Oracle, PostgreSQL 等&#xff0c;仍然承载着大量的关键业务…

React 编译器 RC

&#x1f916; 作者简介&#xff1a;水煮白菜王&#xff0c;一位前端劝退师 &#x1f47b; &#x1f440; 文章专栏&#xff1a; 前端专栏 &#xff0c;记录一下平时在博客写作中&#xff0c;总结出的一些开发技巧和知识归纳总结✍。 感谢支持&#x1f495;&#x1f495;&#…

关于表连接

目录 1.左连接 2.右连接 3.内连接 4.全外连接 5.笛卡尔积 -- 创建表A CREATE TABLE A(PNO VARCHAR2(10) PRIMARY KEY, PAMT NUMBER, A_DATE DATE);-- 向表A插入数据 INSERT INTO A VALUES (01001, 100, TO_DATE(2005-01-01, YYYY-MM-DD)); INSERT INTO A VALUES (010…

【计算机网络】fork()+exec()创建新进程(僵尸进程及孤儿进程)

文章目录 一、基本概念1. fork() 系统调用2. exec() 系列函数 二、典型使用场景1. 创建子进程执行新程序2. 父子进程执行不同代码 三、核心区别与注意事项四、组合使用技巧1. 重定向子进程的输入/输出2. 创建多级子进程 五、常见问题与解决方案僵尸进程&#xff08;Zombie Proc…

Word表格怎样插入自动序号或编号

在Word文档中编辑表格时&#xff0c;经常需要为表格添加序号或编号&#xff0c;可以设置为自动序号或编号&#xff0c;当删除行时&#xff0c;编号会自动变化&#xff0c;不用手工再重新编号。如图所示。 序号数据1数据21300300230030033003004300300 一&#xff0c;建立word表…

无人机仿真环境(3维)附项目git链接

项目概述 随着无人机技术在物流、测绘、应急救援等领域的广泛应用&#xff0c;其自主导航、避障算法、路径规划及多机协同等核心技术的研究需求日益迫切。为降低实地测试成本、提高研发效率&#xff0c;本项目旨在构建一个高精度、可扩展的​​无人机三维虚拟仿真环境​​&…

Python 训练营打卡 Day 30-模块和库的导入

模块和库的导入 1.1标准导入 import mathprint("方式1: 使用 import math") print(f"圆周率π的值: {math.pi}") print(f"2的平方根: {math.sqrt(2)}\n") 1.2从库中导入特定项 from math import pi, sqrtprint("方式2&#xff1a;使用 f…

前端实现图片压缩:基于 HTML5 File API 与 Canvas 的完整方案

在 Web 开发中,处理用户上传的图片时,前端压缩可以有效减少服务器压力并提升上传效率。本文将详细讲解如何通过<input type="file">实现图片上传,结合 Canvas 实现图片压缩,并实时展示压缩前后的图片预览和文件大小对比。 一、核心功能架构 我们将实现以…