深入理解java.lang.String类的不可变性


Posted in Java/Android onJune 27, 2021

1. 字符串 String 的不可变性

什么是不可变类?

这样理解:
        一个对象在创建完成后,不能去改变它的状态,不能改变它的成员变量(如果成员变量包含基本数据类型,那么这个基本数据类型的值不能改变;如果包含引用类型,那么这个引用类型的变量不能指向别的对象)

不可变类只是其实例不能被修改的类。每个实例中包含的所有信息都必须在创建该实例的时候就提供,并且在对象的整个生命周期内固定不变。为了使类不可变,要遵循下面五条规则:

  • 不要提供任何会修改对象状态的方法
  • 保证类不会被扩展。 一般的做法是让这个类称为 final 的,防止子类化,破坏该类的不可变行为
  • 使所有的域都是 final 的
  • 使所有的域都成为私有的。 防止客户端获得访问被域引用的可变对象的权限,并防止客户端直接修改这些对象
  • 确保对于任何可变性组件的互斥访问。 如果类具有指向可变对象的域,则必须确保该类的客户端无法获得指向这些对象的引用

翻阅 API 文档:

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    // value 数组被 final 修饰
    private final char value[];
    ...
}

String 类代表字符串。Java 程序中的所有字符串字面值(如 “abc” )都作为此类的实例来实现。 这些字面值都是直接存储在“方法区”的 字符串常量池

字符串是常量;它们的值在创建之后 不能改变,所以可以共享它们。例如:

String str = "abc";

这时就有人疑惑了:为什么 String 不可变?但我的代码中经常改变 String 啊,如下:

String str = "HELLO";
str = "WORLD";
System.out.println(str);    // WORLD

这样操作,不就是将 “HELLO” 对象改变成了 “WORLD” 对象了吗?

虽然字符串的内容看上去从“HELLO” 变成了“WORLD”,但实际上,这已经是生成了一个新的字符串了:

String str = "HELLO";
System.out.println(str.hashCode());  // 68624562
str = "WORLD";
System.out.println(str.hashCode());  // 82781042

变量 str 前后的 hashCode 值不一样,说明了 str 在改变前后,指向了不同的对象。所以,变量 str 只是指向了不同对象,字符串 “HELLO”对象本身没有被改变。

变量 str 的指向如下图所示(jdk1.8:字符串常量位于堆中):

深入理解java.lang.String类的不可变性

我们也可以使用 javap 命令来查看 class 的常量池:

javap -c -v StringTest.class

执行后,常量池信息如下:

深入理解java.lang.String类的不可变性

从常量池中可以看出,确实有两个字符串对象:HELLO、WORLD

【总结】:一旦一个 String 对象堆中被创建出来,它就无法被修改。而且,String 类的所有 API 方法都没有改变字符串本身的值,都是返回了一个新的字符串对象。

2. String 设计成不可变类的好处

在了解了“String 是不可变”的之后,大家是不是很疑惑:为什么要把 String 设计成不可变的呢?这样做又有什么好处呢?

主要从以下几个角度考虑:

  • 安全可靠性:字符串在 Java 应用程序中应用广泛(存储敏感信息,如:用户名、密码、连接 url、网络连接等);JVM类加载器在加载类的时也广泛地使用它。因此,保护 String 类对于提升整个应用程序的安全性至关重要。
  • 缓存:字符串是使用最广泛的数据结构,大量的字符串的创建是非常耗费资源的。JVM 中专门开辟了一部分空间来存储 Java 字符串,那就是字符串常量池。通过字符串常量池,两个内容相同的字符串变量,可以从池中指向同一个字符串对象,从而节省了关键的内存资源
  • 线程安全:不可变会自动使字符串成为线程安全的,因为当从多个线程访问它们时,它们不会被更改
  • hashcode 缓存:字符串也被广泛地用于哈希实现,如 HashMap、HashTable、HashSet 等。在对这些散列实现进行操作时,经常调用键的hashCode() 方法。不可变性保证了字符串的值不会改变,因此,hashCode() 方法在 String 类中被重写,以方便缓存。这样,在第一次hashCode() 调用期间计算和缓存散列,并从那时起返回相同的值。

3. 面试题

// 生成两个对象:一个在常量池中;一个中堆中,且都是 hello 对象
String s = new String("hello");

那么,下面会生成几个对象呢?

// 只会在字符串常量池中生成一个对象:helloworld。
String s3 = "hello" + "world";

这种字面量用“+”拼接,编译器在编译期间会直接进行优化。

// 这个会生成4个对象。2个在常量池中:hello、world
// 2个在堆中:StringBuilder、helloworld对象
String s = "hello";
String s2 = s + "world";

编译后,使用反编译软件 ------ jad 进行查看:

String s1 = "hell0";
String s2 = (new StringBuilder()).append(s1).append("world").toString();

发现:使用“+”将变量和字面量进行拼接的结果是:将 String 转成了StringBuilder 后,使用其 append() 方法进行处理的

查看 StringBuilder.toString() 方法源码:

@Override
public String toString() {
	// char[] value; value 是 StringBuilder 类的成员变量
    return new String(value, 0, count);
}

最后调用 toString() 方法时,会创建一个 String 对象。这个字符串对象只会在堆中创建,并不会在字符串常量池中创建。所以,会创建4个对象(hello 和 world 会直接在字符串常量池中创建)。

到此这篇关于深入理解java.lang.String类的不可变性的文章就介绍到这了,更多相关java.lang.String不可变性内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Java/Android 相关文章推荐
源码解读Spring-Integration执行过程
Jun 11 Java/Android
JVM入门之类加载与字节码技术(类加载与类的加载器)
Jun 15 Java/Android
SpringBoot2 参数管理实践之入参出参与校验的方式
Jun 16 Java/Android
关于Mybatis中SQL节点的深入解析
Mar 19 Java/Android
SpringBoot2零基础到精通之异常处理与web原生组件注入
Mar 22 Java/Android
Netty分布式客户端处理接入事件handle源码解析
Mar 25 Java/Android
Java十分钟精通进阶适配器模式
Apr 06 Java/Android
Java线程的6种状态与生命周期
May 11 Java/Android
Java实现扫雷游戏详细代码讲解
May 25 Java/Android
springboot为异步任务规划自定义线程池的实现
Jun 14 Java/Android
Java代码规范与质量检测插件SonarLint的使用
Aug 05 Java/Android
Android移动应用开发指南之六种布局详解
Sep 23 Java/Android
springboot拦截器无法注入redisTemplate的解决方法
Java中PriorityQueue实现最小堆和最大堆的用法
探讨Java中的深浅拷贝问题
解决SpringBoot跨域的三种方式
Jun 26 #Java/Android
分析Java中Map的遍历性能问题
Jun 26 #Java/Android
SpringCloud的JPA连接PostgreSql的教程
spring项目中切面及AOP的使用方法
You might like
PHP也可以?成Shell Script
2006/10/09 PHP
模仿OSO的论坛(一)
2006/10/09 PHP
PHP中调用ASP.NET的WebService的代码
2011/04/22 PHP
php自定义函数截取汉字长度
2014/05/15 PHP
ThinkPHP中使用ajax接收json数据的方法
2014/12/18 PHP
Symfony2使用Doctrine进行数据库查询方法实例总结
2016/03/18 PHP
php实现的SSO单点登录系统接入功能示例分析
2016/10/12 PHP
javascript实现网页端解压并查看zip文件
2015/12/15 Javascript
js实现网页收藏功能
2015/12/17 Javascript
jQuery简单操作cookie的插件实例
2016/01/13 Javascript
Vue.js实现简单动态数据处理
2017/02/13 Javascript
js实现扫雷小程序的示例代码
2017/09/27 Javascript
代码实例ajax实现点击加载更多数据图片
2018/10/12 Javascript
基于Vue组件化的日期联动选择器功能的实现代码
2018/11/30 Javascript
vue使用showdown并实现代码区域高亮的示例代码
2019/10/17 Javascript
JS window对象简单操作完整示例
2020/01/14 Javascript
mapboxgl实现带箭头轨迹线的代码
2021/01/04 Javascript
[03:08]迎霜节狂欢!2018年迎霜节珍藏Ⅰ一览
2018/12/25 DOTA
python自动化工具日志查询分析脚本代码实现
2013/11/26 Python
python 读取excel文件生成sql文件实例详解
2017/05/12 Python
Python实现按中文排序的方法示例
2018/04/25 Python
python3字符串操作总结
2019/07/24 Python
python实现邮件自动发送
2019/08/10 Python
python 使用opencv 把视频分割成图片示例
2019/12/12 Python
python 实现rolling和apply函数的向下取值操作
2020/06/08 Python
python实现图书馆抢座(自动预约)功能的示例代码
2020/09/29 Python
HTML5 input placeholder 颜色修改示例
2014/05/30 HTML / CSS
Vision Directa智利眼镜网:框架眼镜、隐形眼镜和名牌太阳眼镜
2016/11/23 全球购物
viagogo波兰票务平台:演唱会、体育比赛、戏剧门票
2018/04/23 全球购物
水上运动奥特莱斯:Wasterports Outlet
2018/08/08 全球购物
大队干部竞选演讲稿
2014/04/28 职场文书
学校与家长安全责任书
2014/07/23 职场文书
销售口号霸气押韵
2015/12/24 职场文书
领导干部学习三严三实心得体会
2016/01/05 职场文书
Redis字典实现、Hash键冲突及渐进式rehash详解
2021/09/04 Redis
前端传参数进行Mybatis调用mysql存储过程执行返回值详解
2022/08/14 MySQL