当我们不再使用某个对象的时候,此对象会被 GC 垃圾回收掉。当然前提是你没有写出内存泄漏的代码。我们也知道如果生成了大量的字符串,会对 GC 造成很大的压力。
但是,如果在编译期间能够确定的字符串,就不会被 GC 垃圾回收掉了。
示例代码
下面,我创建了几个字符串,我关心的字符串是 "walterlv"
,"lindexi"
以及一个当前时间。
于是使用下面的代码来验证:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | using System; using System.Linq; using System.Runtime.CompilerServices; namespace Walterlv.Demo { class Program { static void Main(string[] args) { var table = new ConditionalWeakTable<string, Foo> { {"walterlv", new Foo("吕毅")}, {"lindexi", new Foo("林德熙")}, }; var time = DateTime.Now.ToString("T"); table.Add(time, new Foo("时间")); time = null; Console.WriteLine($"开始个数:{table.Count()}"); GC.Collect(); Console.WriteLine($"剩余个数:{table.Count()}"); } } public class Foo { public string Value { get; } public Foo(string value) => Value = value; } } |
---|
"walterlv"
和 "lindexi"
是在编译期间能够完全确定的字符串,而当前时间字符串我们都知道是编译期间不能确定的字符串。
在 GC 收集之前和之后,ConditionalWeakTable
中的对象数量从三个降到了两个。
并没有清除成 0 个,说明字符串现在仍然是被引用着的。
那被什么引用着呢?是字符串暂存池。要理解字符串暂存池,可以阅读我的另一篇博客:
- .NET/C# 的字符串暂存池
另外,即便设置了 CompilationRelaxations.NoStringInterning
,编译期间能确定的字符串在上述代码中也是不会被垃圾回收的。
参考资料
- c# - Strings and Garbage Collection - Stack Overflow
本文会经常更新,请阅读原文: https://blog.walterlv.com/post/compile-time-strings-are-in-the-string-intern-pool.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 吕毅 (包含链接: https://blog.walterlv.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 ([email protected]) 。