java 为什么 String 在 java 中是不可变的?

简介: 本文探讨了Java中String为何设计为不可变类型,从字符串池的高效利用、哈希码缓存、支持其他对象的安全使用、增强安全性以及线程安全等方面阐述了不可变性的优势。文中还通过具体代码示例解释了这些优点的实际应用。

为什么 String 在 java 中是不可变的?

String 在 java 中是不可变的,一个不可变类意味着它的实例在创建之后就不可修改,实例的所有属性在创建时初始化,之后无法对这些属性进行修改。不可变类型有着许多的优点,这篇文章总结了 为什么 String 被设计成不可变的,文章将从内存、同步和数据结构的角度说明不变性概念。

1 字符串池的需要

字符串池是存在于 Java方法区 的一个特殊内存区域,当需要创建的目标字符串在字符串池中已经存在,那么字符串池中的字符串引用就会返回并赋值给目标字符串,而不是创建一个新的对象。

如下的代码只会在堆中创建一个对象:

代码解读

复制代码

String string1 = "abc";
String string2 = "abc";

图形表示:

如果 String 是可变的,一个引用改变字符串的值将会导致另一个引用在获取值时得到错误的值。(译者:另一个引用并未对字符串做修改,当他再次取值时字符串的值却与上次取的不同!)。

2 用作缓存时的 hashcode

字符串的哈希值在 java 中是被频繁使用到的。举个例子,在 HashMap 或 HashSet 中,String 的不可变性保证了字符串 hashcode 的一致性,所以在进行缓存时无需担心字符串变化,这意味着,不需要在字符串每次被使用到时都计算其 hashcode 来保证一致性,这样更加高效。

在 String 类源码中有如下的代码:

代码解读

复制代码

    /** Cache the hash code for the string */
    private int hash; // Default to 0

3 便利其它对象的使用

具体化来说,考虑如下代码:

代码解读

复制代码

HashSet<String> set = new HashSet<String>();
set.add(new String("a"));
set.add(new String("b"));
set.add(new String("c"));
for (String s : set){
    a.value = "a";
}

在这个例子中,如果 String 是可变的,那么它的 value 可以被改变,但这违反了 Set 的设计原则(Set 中的元素是不能重复的),当然,上面的例子仅仅为了表明意图,String 类中并不存在 value 成员。

4 安全性

String 作为参数在 java 中广泛使用,例如网络连接,打开文件,等。如果 String 被设计为可变的,那么一个网络连接或者文件操作将会被改变,这可能留下非常严重的安全隐患。方法调用的预期结果是成功连接到设备,事实可能是并没有连接。可变的字符串在反射中也会导致安全问题,因为参数是字符串。

代码示例:

代码解读

复制代码

boolean connect(String str){
    if (!isSecure(str)){
        throw new SecureException();
    }
    //在方法调用前如果 str 的值被其它引用对象改变,就会出现问题。
    caseProblem(str);
}

5 不可变对象是天生线程安全的

因为不可变对象不会被改变,它们可以在多个线程间自由访问。这样就无需对存取进行同步。

总结,String 被设计为 final 的原因是 效率安全,通常情况下这也是为什么不可变对象在许多设计中会成为首选的原因。


转载来源:https://juejin.cn/post/6844903537202184206

相关文章
|
2月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
308 5
|
4月前
|
存储 SQL 缓存
Java字符串处理:String、StringBuilder与StringBuffer
本文深入解析Java中String、StringBuilder和StringBuffer的核心区别与使用场景。涵盖字符串不可变性、常量池、intern方法、可变字符串构建器的扩容机制及线程安全实现。通过性能测试对比三者差异,并提供最佳实践与高频面试问题解析,助你掌握Java字符串处理精髓。
|
5月前
|
自然语言处理 Java Apache
在Java中将String字符串转换为算术表达式并计算
具体的实现逻辑需要填写在 `Tokenizer`和 `ExpressionParser`类中,这里只提供了大概的框架。在实际实现时 `Tokenizer`应该提供分词逻辑,把输入的字符串转换成Token序列。而 `ExpressionParser`应当通过递归下降的方式依次解析
348 14
|
Java 索引
java基础(13)String类
本文介绍了Java中String类的多种操作方法,包括字符串拼接、获取长度、去除空格、替换、截取、分割、比较和查找字符等。
177 0
java基础(13)String类
|
9月前
|
缓存 安全 Java
《从头开始学java,一天一个知识点》之:字符串处理:String类的核心API
🌱 **《字符串处理:String类的核心API》一分钟速通!** 本文快速介绍Java中String类的3个高频API:`substring`、`indexOf`和`split`,并通过代码示例展示其用法。重点提示:`substring`的结束索引不包含该位置,`split`支持正则表达式。进一步探讨了String不可变性的高效设计原理及企业级编码规范,如避免使用`new String()`、拼接时使用`StringBuilder`等。最后通过互动解密游戏帮助读者巩固知识。 (上一篇:《多维数组与常见操作》 | 下一篇预告:《输入与输出:Scanner与System类》)
258 11
|
9月前
|
Java
课时14:Java数据类型划分(初见String类)
课时14介绍Java数据类型,重点初见String类。通过三个范例讲解:观察String型变量、&quot;+&quot;操作符的使用问题及转义字符的应用。String不是基本数据类型而是引用类型,但使用方式类似基本类型。课程涵盖字符串连接、数学运算与字符串混合使用时的注意事项以及常用转义字符的用法。
276 9
|
Java 测试技术 开发者
Java零基础-indexOf(String str)详解!
【10月更文挑战第14天】Java零基础教学篇,手把手实践教学!
319 65
|
12月前
|
存储 JavaScript Java
Java 中的 String Pool 简介
本文介绍了 Java 中 String 对象及其存储机制 String Pool 的基本概念,包括字符串引用、构造方法中的内存分配、字符串文字与对象的区别、手工引用、垃圾清理、性能优化,以及 Java 9 中的压缩字符串特性。文章详细解析了 String 对象的初始化、内存使用及优化方法,帮助开发者更好地理解和使用 Java 中的字符串。
183 2
Java 中的 String Pool 简介
|
JSON Java 关系型数据库
Java更新数据库报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
在Java中,使用mybatis-plus更新实体类对象到mysql,其中一个字段对应数据库中json数据类型,更新时报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
1264 4
Java更新数据库报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.