当前位置: 首页 > ds >正文

30、不是说字符串是不可变的吗,string s=“abc“;s=“123“不就是变了吗?

一、核心概念澄清:不可变性的真实含义

1、不可变性的定义

字符串不可变性指对象内容不可修改,而非变量不可修改。
类比:

  • 不可变字符串 = 装在密封信封里的信纸(内容不可更改)
  • 变量赋值 = 更换信封的指向(从A信封换成B信封)

2、代码示例的真相

string s = "abc";   // 创建信封A,指向"abc"信纸
s = "123";          // 创建新信封B,指向"123"信纸(A仍存在)

关键点:s是引用(信封),字符串是对象(信纸),修改的是引用指向而非对象内容。

二、底层实现机制:内存中的真相

1、对象层面的不可变性

  • 字符串对象在内存中存储为只读字符数组,无提供修改内容的公开方法。
  • 反例:若尝试直接修改会编译错误(C#):
string s = "abc";
// s[0] = '1'; // 编译错误:无法修改只读字段

2、引用层面的可变性

变量是托管堆上的指针,可自由指向不同对象。
类比:

  • 字符串对象 = 房子
  • 变量 = 房产证(可更换指向不同房子)

三、不可变性的工程价值

1、线程安全的基石

多线程环境下无需同步:

string globalStr = "shared";
// 线程1和线程2同时读取globalStr无需加锁

2、哈希键的稳定性

字典/哈希表依赖对象哈希值不变:

var dict = new Dictionary<string, int> { ["key"] = 1 };
string key = "key";
key = "new"; // 不影响dict["key"]的查找

3、内存池优化

字符串驻留(String Interning)复用相同字符串对象:

string s1 = "hello";
string s2 = "hello"; // 可能指向同一内存地址

四、常见误解的根源

1、变量重新赋值的表象

错误类比:将s = "123"等同于修改s本身,实际是修改s的指向。
对比可变对象:

StringBuilder sb = new StringBuilder("abc");
sb.Append("123"); // 真正修改对象内容

2、语言差异的混淆

  • C++的std::string是可变的,但C#/Java的string设计不同。
  • 性能权衡:不可变性牺牲部分修改性能,换取安全性和复用性。

五、验证不可变性的方法

1、引用关系验证

string s1 = "test";
string s2 = s1;
s1 = "new";
Console.WriteLine(s2); // 输出"test",证明s1修改不影响s2

2、对象内容验证

使用String.GetHashCode():

string s = "abc";
int hash1 = s.GetHashCode();
s = "123";
int hash2 = s.GetHashCode(); // hash1 != hash2

六、工程实践建议

1、高频修改场景的替代方案

使用StringBuilder(C#)

var sb = new StringBuilder("abc");
sb.Append("123"); // 高效拼接

2、防御性编程技巧

避免返回字符串参数的引用:

// 错误示例(可能暴露内部可变性)
public string GetData() => _internalString; // 正确做法(返回新对象)
public string GetData() => new string(_internalString);

七、总结

字符串不可变性指的是对象内容不可修改,而非变量不可修改。s='123’实际是变量s重新指向新对象,原字符串’abc’仍存在于内存中。这种设计保证了线程安全、哈希键稳定性和内存池复用。需要修改字符串时,应使用StringBuilder等可变类型替代。验证不可变性可通过观察变量修改不影响其他引用或对象哈希值变化。

在这里插入图片描述

http://www.xdnf.cn/news/2500.html

相关文章:

  • HNUST湖南科技大学-嵌入式考试选择题题库(109道纠正详解版)
  • 电镀废水资源化利用的工艺介绍
  • 备份服务器,备份服务器数据有哪些方法可以实现?
  • Spark学习全总结
  • 多用户商城系统的三种形式:综合性、垂直型、社交型
  • MPI Code for Ghost Data Exchange in 3D Domain Decomposition with Multi-GPUs
  • ALTER TABLE 删除DROP表列的报错: 因为有一个或多个对象访问此列
  • 多通道示波器测量系统的工程实践与技术演进
  • Flink 时态维度表 Join 与缓存机制实战
  • (done) 吴恩达版提示词工程 8. 聊天机器人 (聊天格式设计,上下文内容,点餐机器人)
  • ppt流程图怎么?ppt流程图模板大全
  • 【C语言操作符详解(一)】--进制转换,原反补码,移位操作符,位操作符,逗号表达式,下标访问及函数调用操作符
  • 自动驾驶(ADAS)领域常用数据集介绍
  • 学习insightface 的人脸识别
  • 企业如何构建一个全面的Web安全防护体系
  • PDF处理控件Aspose.PDF指南:如何使用 C# 在 PDF 中搜索
  • STM32 定时器TIM
  • 重塑编程体验边界:明基RD280U显示器深度体验
  • redis常用集合操作命令
  • C#如何正确的停止一个多线程Task?CancellationTokenSource 的用法。
  • 泰迪杯实战案例超深度解析:运输车辆安全驾驶行为分析与安全评价系统设计
  • 基于边缘人工智能的AI无人机-更高效更安全的飞行任务执行
  • macos下mysql 5.7/8.0版本切换
  • 如何修复Chrome浏览器的“无法连接到互联网”错误
  • 14、服务端组件:未来魔法预览——React 19 RSC实践
  • 《代码整洁之道》第10章 类 - 笔记
  • 谢飞机的Java面试之旅:从Spring Boot到Kubernetes的挑战
  • 用Python做有趣的AI项目 3:黑白图像自动上色(AI 上色器)
  • 数智读书笔记系列031《HIS内核设计之道——医院信息系统规划设计系统思维》书籍简介与读书笔记
  • 【读写视频】MATLAB详细代码