C# 字符串不可变性 + 字符串驻留池原理

一、字符串不可变性(String Immutability)

1. 定义

C# 中 string 是不可变(只读)引用类型一旦字符串在内存中创建,就永远不能被修改 ,任何 "修改" 操作都不会改动原字符串 ,而是新建一个字符串

2. 为什么不可变
  1. 线程安全:只读,多线程同时读不用加锁

  2. 支持驻留池:相同文本复用同一块内存,不怕被篡改

  3. 简化 GC、缓存、哈希设计

    cs 复制代码
    string s = "abc";
    s.ToUpper();
    Console.WriteLine(s); // 还是 abc,原字符串没变

    ToUpper() 没有改原 s ,而是返回了一个全新字符串对象

再看拼接:

cs 复制代码
string a = "123";
a += "456";

底层:

  • 不修改原 "123"
  • 重新分配内存,创建 "123456" 新字符串
  • a 引用指向新对象,旧对象等待 GC
3. 不可变带来的问题

频繁拼接字符串(循环里 +=)会不断生成新字符串、大量创建临时对象、触发 GC 。解决方案:StringBuilder StringBuilder可变的,内部维护字符缓冲区,原地修改,不频繁新建对象。

二、字符串驻留池(String Intern Pool)

1. 是什么

CLR 维护的一个全局字符串缓存池 ,目的:复用相同内容的字符串实例,节约内存、减少重复分配

核心规则:内容相同的字符串,在驻留池中只存一份,多个引用指向同一个内存地址

2. 驻留池分类
  1. 编译期驻留(常量字符串)
  2. 运行期手动驻留(string.Intern()

三、编译期驻留

原理

代码里双引号直接写的字面量字符串,编译时 CLR 会:

示例证明地址相同

cs 复制代码
string s1 = "hello";
string s2 = "hello";

// 值相等
Console.WriteLine(s1 == s2);      
// 引用地址也相等 同一个对象
Console.WriteLine(object.ReferenceEquals(s1, s2)); 

输出都是 True👉 s1s2 指向堆上同一个字符串实例

不进入驻留池的情况

运行时动态拼接、new 出来的字符串,默认不驻留

cs 复制代码
string s1 = "hello";
string s2 = "hel" + "lo";   // 编译器优化,还是字面量,会驻留
string s3 = new string("hello".ToCharArray()); 

Console.WriteLine(object.ReferenceEquals(s1, s3)); // False

s3 是运行时构造,不在驻留池,是新对象。

四、运行期手动驻留:string.Intern ()

用法

cs 复制代码
string s3 = new string("hello".ToCharArray());
string internStr = string.Intern(s3);

// 现在和 s1 指向同一个驻留池实例
Console.WriteLine(object.ReferenceEquals(s1, internStr)); // True

Intern 原理

  1. 拿字符串内容去驻留池查找
  2. 找到:返回池里已有实例引用
  3. 没找到:把当前字符串加入驻留池,返回引用

适用场景:大量重复动态字符串(如日志、解析文本),手动驻留省内存。

五、驻留池存在哪里

  • .NET Framework:进程级全局驻留池,程序运行一直存在
  • .NET Core/.NET 5+:每个 AppDomain 独立驻留池 驻留池里的字符串生命周期很长,不容易被 GC 回收。

六、不可变性 + 驻留池 关联关系

  • 正因为字符串不可变,才敢做驻留池
  • 如果字符串可修改,一个引用改了内容,所有指向它的变量都会被篡改,完全乱套
  • 不可变 + 驻留池 = 安全复用内存

七、总结

  • 字符串不可改,一改建新串
  • 字面量进驻留池,同内容共用一块内存
  • 动态拼接默认不驻留,地址不同
  • string.Intern 手动入池,复用实例省内存
  • 频繁拼接用 StringBuilder,避免大量生成新字符串
相关推荐
周杰伦fans1 小时前
C# 踩坑 CS8370:Switch Expression 在 C# 7.3 不可用及三种解决方案
c#
z落落3 小时前
C# ToCharArray + foreach遍历 + String与StringBuilder
开发语言·c#
xiaoshuaishuai84 小时前
C# AvaloniaUI动态显示图片
开发语言·c#
csdn_aspnet7 小时前
EasyModbus 与 C# 集成
c#·modbus·easymodbus
JaydenAI9 小时前
[MAF预定义ChatClient中间件-06]利用ImageGeneratingChatClient开发专业图片生成Agent
ai·c#·agent·agent管道·chatclient中间件·chatclient管道
csdn_aspnet10 小时前
Modbus TCP C# 客户端程序
服务器·网络·tcp/ip·c#
0x000711 小时前
译 Anders Hejlsberg 谈 C# 与 .NET
开发语言·c#·.net
Xin_ye1008611 小时前
C# 零基础到精通教程 - 第十七章:前端集成——Blazor 基础
开发语言·c#
daopuyun11 小时前
《C#语言源代码漏洞测试规范》解读,如何依据GB/T 34946-2017标准建立代码测试技术体系
c#·代码测试·源代码安全检测