【面试】Java 之 String 系列 -- String 为什么不可变?

devtools/2025/3/1 18:58:48/

        在 Java 编程中,String 类是一个使用频率极高的类。而 String 对象具有不可变的特性,这一特性在 Java 设计中有着重要的意义。本文将深入探讨 String 不可变的含义、原因以及带来的好处。

一、String 不可变的含义

1. 概念解释

所谓 String 不可变,指的是一旦一个 String 对象被创建,它的内容(即字符序列)就不能被改变。在 Java 里,String 类被设计为 final 类,并且其内部用于存储字符序列的 value 数组也是 private 和 final 的。以下是 String 类中部分相关代码:

java">public final class Stringimplements java.io.Serializable, Comparable<String>, CharSequence {/** The value is used for character storage. */private final char value[];// 其他代码...
}

从代码中可以看出,value 数组被 final 修饰,这意味着一旦数组被初始化,其引用就不能再指向其他数组。而 private 修饰符则保证了外部无法直接访问和修改这个数组。

2. 示例说明

下面通过一个简单的示例来直观感受 String 的不可变性:

java">public class StringImmutabilityExample {public static void main(String[] args) {String str = "Hello";str = str + " World";System.out.println(str);}
}

就像下面这个图示一样: 

在这个例子中,我们可能会认为 str 的内容从 "Hello" 变成了 "Hello World"。但实际上,str 最初指向的 "Hello" 对象并没有改变,当执行 str = str + " World" 时,Java 会创建一个新的 String 对象 "Hello World",然后让 str 引用这个新对象,而原来的 "Hello" 对象仍然存在于内存中。

二、String 不可变的原因

1. 安全性

String 在 Java 中广泛用于存储敏感信息,如用户名、密码、数据库连接信息等。如果 String 是可变的,那么这些敏感信息就可能被恶意修改,从而引发安全问题。例如,在多线程环境下,如果一个线程正在使用一个 String 对象存储的密码进行验证,而另一个线程同时修改了这个 String 对象的内容,那么验证结果就会变得不可靠。

2. 缓存哈希码

String 类重写了 hashCode() 方法,并且 String 对象的哈希码是在对象创建时就计算好并缓存起来的。因为 String 不可变,所以其哈希码不会改变,这样在使用 String 作为哈希表(如 HashMapHashSet)的键时,就可以避免重复计算哈希码,提高了性能。以下是 String 类中 hashCode() 方法的部分代码:

java">public final class Stringimplements java.io.Serializable, Comparable<String>, CharSequence {private int hash; // Default to 0public int hashCode() {int h = hash;if (h == 0 && value.length > 0) {char val[] = value;for (int i = 0; i < value.length; i++) {h = 31 * h + val[i];}hash = h;}return h;}// 其他代码...
}

可以看到,如果 String 可变,那么哈希码就可能会因为内容的改变而改变,这将破坏哈希表的正常工作。

3. 便于字符串常量池的实现

Java 中的字符串常量池是一种特殊的内存区域,用于存储字符串常量。当使用双引号声明一个字符串时,Java 会首先在字符串常量池中查找是否已经存在相同内容的字符串,如果存在,则直接返回该字符串的引用;如果不存在,则在常量池中创建一个新的字符串对象。String 的不可变性保证了常量池中的字符串可以被安全地共享,避免了因内容改变而导致的混乱。例如:

java">String str1 = "Hello";
String str2 = "Hello";
System.out.println(str1 == str2); // 输出 true

在这个例子中,str1 和 str2 都指向字符串常量池中的同一个 "Hello" 对象。

三、String 不可变带来的好处

1. 线程安全

由于 String 不可变,所以在多线程环境下,多个线程可以同时访问同一个 String 对象,而不需要担心数据被修改的问题。这使得 String 成为了线程安全的类,开发者可以放心地在多线程程序中使用 String 对象,无需额外的同步机制。

2. 性能优化

正如前面提到的,String 的不可变性使得哈希码可以被缓存,这在使用 String 作为哈希表的键时可以显著提高性能。此外,由于 String 对象可以在字符串常量池中共享,减少了内存的使用,也提高了垃圾回收的效率。

3. 代码可读性和可维护性

String 的不可变性使得代码更加易于理解和维护。因为开发者不需要担心 String 对象的内容会在程序运行过程中被意外修改,所以可以更专注于业务逻辑的实现。

四、利用反射改变 String 的字符数据

        虽然我不能不能改变字符串,但是我们可以修改字符串的字符,Java 的反射机制允许我们在运行时检查和修改类的属性、方法等。通过反射,我们可以绕过 private 修饰符的限制,访问并修改 String 对象内部的 value 数组。

以下是一个示例代码:

java">package org.example.a;import java.lang.reflect.Field;public class Demo {public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {String s = "1234";System.out.println("改变前:s=" + s);// 获取 String 类的 value 属性Field f = s.getClass().getDeclaredField("value");// 设置该属性可访问,绕过 private 限制f.setAccessible(true);// 修改 value 数组为新的字符数组f.set(s, new char[]{'a', 'b', 'c'});System.out.println("改变后:s=" + s);}
}

代码解释

  • Field f = s.getClass().getDeclaredField("value");:通过 getClass() 方法获取 s 对象的 Class 对象,然后使用 getDeclaredField("value") 方法获取 String 类中名为 value 的属性。
  • f.setAccessible(true);:将 value 属性的可访问性设置为 true,这样就可以绕过 private 修饰符的限制,对其进行访问和修改。
  • f.set(s, new char[]{'a', 'b', 'c'});:将 s 对象的 value 属性设置为新的字符数组 {'a', 'b', 'c'}

执行结果

改变前:s=1234
改变后:s=abc

从执行结果可以看出,我们成功地通过反射修改了 String 对象的字符数据。

3. 这种做法的风险和注意事项

虽然通过反射可以改变 String 的字符数据,但这种做法并不推荐在实际开发中使用,原因如下:

  • 破坏不可变性原则String 的不可变性是 Java 语言设计的重要特性之一,许多系统和库都依赖于这一特性。使用反射修改 String 会破坏这种不可变性,可能导致程序出现难以调试的错误。
  • 安全问题:如果恶意代码利用反射修改 String 对象,可能会导致安全漏洞,比如篡改敏感信息等。
  • 兼容性问题:不同的 Java 版本或虚拟机实现可能对反射操作的支持有所不同,使用反射修改 String 可能会导致兼容性问题。

http://www.ppmy.cn/devtools/163699.html

相关文章

使用torch.compile进行CPU优化

在PyTorch中&#xff0c;使用torch.compile可以自动地将模型转换成优化的执行代码&#xff0c;这对于提升模型在CPU上的运行效率尤其有用。torch.compile是基于TorchDynamo实现的&#xff0c;它可以将Python代码转换为高效的TorchScript代码。这对于那些在CPU上运行的大型模型尤…

AI人工智能机器学习之聚类分析

1、概要 本篇学习AI人工智能机器学习之聚类分析&#xff0c;以KMeans、AgglomerativeClustering、DBSCAN为例&#xff0c;从代码层面讲述机器学习中的聚类分析。 2、聚类分析 - 简介 聚类分析是一种无监督学习的方法&#xff0c;用于将数据集中的样本划分为不同的组&#xff…

DeepSeek 提示词:常见指令类型

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;精通Java编…

智能家居的二次进化:当三维设计遇见场景芯片

在2025年第二届长江经济带水域经济博览会上&#xff0c;广州凡拓数字创意科技股份有限公司&#xff08;简称“凡拓数创”&#xff09;凭借其“AI模型数字孪生”技术组合惊艳亮相&#xff0c;展示了智慧水厂数字孪生运营管理解决方案的硬核实力。而这家深耕数字孪生与AI技术20余…

【Golang学习之旅】Go-zero + Gen:如何使用 Gen 提升 Go 开发效率

文章目录 前言一、Go-zero简介二、Gen工具简介2.1 Gen的功能与特点2.2 Gen的工作原理 三、Go-zero Gen&#xff1a;结合的优势3.1为什么选择Go-zero与Gen3.2 Gen的代码生成与Go-zero的结合点 四、实际案例&#xff1a;Go-zero Gen的应用4.1 构建一个用户管理系统4.2 定义Gen配…

Ubuntu 下 nginx-1.24.0 源码分析 - ngx_init_cycle 函数 - 详解(11)

详解&#xff08;11&#xff09; 初始化配置解析上下文 senv environ;ngx_memzero(&conf, sizeof(ngx_conf_t));/* STUB: init array ? */conf.args ngx_array_create(pool, 10, sizeof(ngx_str_t));if (conf.args NULL) {ngx_destroy_pool(pool);return NULL;}conf.te…

005 公网访问 docker rocketmq

文章目录 创建自定义网络创建NameServer容器创建Broker容器正式开始启动 Nameserver 容器启动 Broker 容器并关联 Nameserverdocker exec -it rmqbroker vi /etc/rocketmq/broker.conf检查 namesrv 解析检查 Broker 注册状态Nameserver 日志Broker 日志检查容器日志手动指定 Br…

硬编码(三)经典变长指令一

我们在前两节的硬编码中学习了定长指令&#xff0c;接下来学习变长指令。学习变长指令要求我们学会查表&#xff1a;intel手册卷2A和2B部分 对于定长指令&#xff0c;我们通过opcode便可知该指令的长度&#xff0c;但是对于变长指令却是不可知的。变长指令长度由opcode&#x…