解析 String 的不可变性

2023-10-21 09:59
文章标签 string 解析 可变性

本文主要是介绍解析 String 的不可变性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问: Java 中的 String 类能不能被继承?为什么 String 有不可变性?这种设计有什么好处?


一、String 的不可变性

String 是 final 类型,final 类不能被继承。

String 是不可变的,当修改已有字符串的值的时候,(如将 str = "abc" 修改为 “abcdef”),不是在原内存地址上修改数据,而是将原引用重新指向新对象、新地址。为什么不可变呢?

在 jdk 1.8 的源码中,java.lang.String 类的起头几行为:

public final class Stringimplements java.io.Serializable, Comparable<String>, CharSequence {/** The value is used for character storage. */private final char value[];/** Cache the hash code for the string */private int hash; // Default to 0......
}

String 本质是个 char 数组,用成员变量 value[] 存储。成员变量 value[] 用 final 关键字修饰,表示该字段创建后引用地址不可变;访问权限为 private,表示外部无法访问,成员变量 hash 也是如此。而且 String 没有对外提供可以修改这两个私有属性的方法。

String 类本身声明为 final,不可被继承,即无法通过扩展来绕过限制。

另外一点就是,构造对象时,成员变量使用 深拷贝 来初始化,而不是直接复制,以防止输入对象被修改。例如,当传入可变数组 value[] 进行初始化时:

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

二、这样设计的原因及好处:

其实好处就是原因,String 设计出不可变,主要是从性能和安全两方面考虑。

1. 字符串常量池

Java 中的字符串常量池(String pool)的存在就是为了性能优化。

字符串常量池是 Java 堆内存中一个特殊的存储区域。当创建一个 String 对象时,首先会去字符串常量池中查找,如果找到,直接返回该字符串的引用,不会创建新的对象。这样做能够减少 JVM 的内存开销,提高性能。如以下代码:

String str1 = "abc";
String str2 = "abc";

此时引用 str1 和 str2 指向常量池中的同一个对象 “abc”,如下图所示:

String 常量池

如果 String 是可变类,那么引用 str1 对 String 对象的修改,会直接导致引用 str2 获取错误的值。所以,如果字符串是可变的,那么字符串常量池就没有存在的意义了。

2. 缓存 HashCode

Java 中经常会用到字符串的哈希码(hashCode)。在 String 类中,有以下代码:

/** Cache the hash code for the string */
private int hash; 

可见,成员变量 hash 中保存了一个 String 对象的 hashCode。因为 String 类不可变,所以一旦对象被创建,该 hash 值便被缓存起来,无法改变了。以后每次使用该对象的 hashCode 的时候,无需重新计算,直接返回即可。这使得字符串很适合作为 HashMap 中的 key,效率大大提高。

3. 多线程安全

多线程中,可变类型的值很可能在其它线程中被改变,造成不可预期的后果。而不可变的 String 可以自由在多个线程之间共享,不需要做任何同步处理。

所以,String 类的不可变性是一个精良的设计。


如有问题,欢迎交流~

博客原文:Deecyn’s Blog

这篇关于解析 String 的不可变性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/253634

相关文章

深度解析Spring Security 中的 SecurityFilterChain核心功能

《深度解析SpringSecurity中的SecurityFilterChain核心功能》SecurityFilterChain通过组件化配置、类型安全路径匹配、多链协同三大特性,重构了Spri... 目录Spring Security 中的SecurityFilterChain深度解析一、Security

全面解析Golang 中的 Gorilla CORS 中间件正确用法

《全面解析Golang中的GorillaCORS中间件正确用法》Golang中使用gorilla/mux路由器配合rs/cors中间件库可以优雅地解决这个问题,然而,很多人刚开始使用时会遇到配... 目录如何让 golang 中的 Gorilla CORS 中间件正确工作一、基础依赖二、错误用法(很多人一开

Mysql中设计数据表的过程解析

《Mysql中设计数据表的过程解析》数据库约束通过NOTNULL、UNIQUE、DEFAULT、主键和外键等规则保障数据完整性,自动校验数据,减少人工错误,提升数据一致性和业务逻辑严谨性,本文介绍My... 目录1.引言2.NOT NULL——制定某列不可以存储NULL值2.UNIQUE——保证某一列的每一

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

MySQL CTE (Common Table Expressions)示例全解析

《MySQLCTE(CommonTableExpressions)示例全解析》MySQL8.0引入CTE,支持递归查询,可创建临时命名结果集,提升复杂查询的可读性与维护性,适用于层次结构数据处... 目录基本语法CTE 主要特点非递归 CTE简单 CTE 示例多 CTE 示例递归 CTE基本递归 CTE 结

Java获取当前时间String类型和Date类型方式

《Java获取当前时间String类型和Date类型方式》:本文主要介绍Java获取当前时间String类型和Date类型方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录Java获取当前时间String和Date类型String类型和Date类型输出结果总结Java获取

Spring Boot 3.x 中 WebClient 示例详解析

《SpringBoot3.x中WebClient示例详解析》SpringBoot3.x中WebClient是响应式HTTP客户端,替代RestTemplate,支持异步非阻塞请求,涵盖GET... 目录Spring Boot 3.x 中 WebClient 全面详解及示例1. WebClient 简介2.

在MySQL中实现冷热数据分离的方法及使用场景底层原理解析

《在MySQL中实现冷热数据分离的方法及使用场景底层原理解析》MySQL冷热数据分离通过分表/分区策略、数据归档和索引优化,将频繁访问的热数据与冷数据分开存储,提升查询效率并降低存储成本,适用于高并发... 目录实现冷热数据分离1. 分表策略2. 使用分区表3. 数据归档与迁移在mysql中实现冷热数据分

C#解析JSON数据全攻略指南

《C#解析JSON数据全攻略指南》这篇文章主要为大家详细介绍了使用C#解析JSON数据全攻略指南,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、为什么jsON是C#开发必修课?二、四步搞定网络JSON数据1. 获取数据 - HttpClient最佳实践2. 动态解析 - 快速

Spring Boot3.0新特性全面解析与应用实战

《SpringBoot3.0新特性全面解析与应用实战》SpringBoot3.0作为Spring生态系统的一个重要里程碑,带来了众多令人兴奋的新特性和改进,本文将深入解析SpringBoot3.0的... 目录核心变化概览Java版本要求提升迁移至Jakarta EE重要新特性详解1. Native Ima