.NET中的字符串駐留池介紹
在.NET中,對于相同的字符串,.NET會將它們指向同一個(gè)地址,它們是相同的實(shí)例。.NET中的字符串并不會更新,當(dāng)更改一個(gè)字符串變量時(shí),由于字符串的不可變性,.NET實(shí)際上是新創(chuàng)建一個(gè)字符串,而將變量地址指向新創(chuàng)建的字符串地址。
看下面的一個(gè)例子:
using System;
namespace ConsoleApp2
{
class Program
{
static void Main(string[] args)
{
string str1 = "hello";
string str2 = "hello";
bool tf = object.ReferenceEquals(str1, str2);
Console.WriteLine(tf);
Console.ReadKey();
}
}
}程序執(zhí)行結(jié)果

從執(zhí)行結(jié)果我們可以得出結(jié)論:str1和str2指向同一個(gè)內(nèi)存對象,它們是同一個(gè)實(shí)例。
在.NET中,CLR默默地維護(hù)了一個(gè)叫做駐留池(Intern Pool)的表。這個(gè)表記錄了所有在代碼中使用字面量聲明的字符串實(shí)例的引用。這說明使用字面量聲明的字符串會進(jìn)入駐留池,而其他方式聲明的字符串則不會進(jìn)入駐留池,也就不會自動享受到CLR防止字符串冗余的機(jī)制的好處了。
看下面一個(gè)例子
using System;
using System.Text;
namespace ConsoleApp2
{
class Program
{
static void Main(string[] args)
{
StringBuilder sb = new StringBuilder();
sb.Append("he").Append("llo");
string str1 = "hello";
string str2 = sb.ToString();
bool tf = Object.ReferenceEquals(str1, str2);
Console.WriteLine(tf);
bool tf1 = str1 == str2;
Console.WriteLine($"str1和str2的內(nèi)容是否相同:{tf1}");
Console.ReadKey();
}
}
}程序執(zhí)行結(jié)果

這里輸出了False。雖然str1和str2是相同的字符串,但是由于str2不是通過字面量的方式聲明的,CLR在為ToString()返回值分配內(nèi)存時(shí),并不會到駐留池中去檢查是否有值為"hello"的字符串已經(jīng)存在了,所以也不會讓str2指向駐留池中的對象。
如果希望強(qiáng)制CLR檢查駐留池,以避免冗余的字符串副本,String類的設(shè)計(jì)者提供了一個(gè)名為Intern的類方法。下面是該方法的一個(gè)實(shí)例
using System;
using System.Text;
namespace ConsoleApp2
{
class Program
{
static void Main(string[] args)
{
StringBuilder sb = new StringBuilder();
sb.Append("he").Append("llo");
string str1 = "hello";
// 在這里強(qiáng)制檢查字符串駐留池
string str2 = string.Intern(sb.ToString());
bool tf = Object.ReferenceEquals(str1, str2);
// 輸出True,因?yàn)闄z查駐留池時(shí),發(fā)現(xiàn)字符串已經(jīng)存在
Console.WriteLine(tf);
bool tf1 = str1 == str2;
Console.WriteLine($"str1和str2的內(nèi)容是否相同:{tf1}");
Console.ReadKey();
}
}
}程序執(zhí)行結(jié)果

Intern方法接受一個(gè)字符串作為參數(shù),它會在駐留池中檢查是否已經(jīng)存在參數(shù)所表示的字符串。如果存在,則返回那個(gè)駐留池中的字符串的引用;否則向駐留池中加入一個(gè)新的表示相同值的字符串,并返回這個(gè)字符串的引用。
不過要注意的是,就算Intern方法在駐留池中找到了相同值的字符串,也不能讓您省卻一次字符串內(nèi)存分配的操作,因?yàn)樽鳛閰?shù)的字符串已經(jīng)被分配了一次內(nèi)存了。使用Intern方法的好處在于,如果Intern方法在駐留池中找到了相同值的字符串,此時(shí)雖然在內(nèi)存中存在兩份該字符串的副本(一份是參數(shù),一份是駐留池中的),但是隨著時(shí)間的流逝,參數(shù)所引用的那個(gè)副本會被垃圾回收掉,這樣對于該字符串內(nèi)存中就不存在冗余了。
當(dāng)你的程序中存在某個(gè)方法,可以根據(jù)不同的上下文環(huán)境創(chuàng)建并返回一個(gè)很長的字符串,而在程序運(yùn)行的過程中它會經(jīng)常返回同樣的字符串時(shí),你可能就要考慮使用Intern方法來提高內(nèi)存的使用率了。不過同樣值得注意的是,使用Intern方法讓一個(gè)字符串存留于駐留池中也有一個(gè)副作用:即使已經(jīng)不存在任何其它引用指向駐留池中的字符串了,這個(gè)字符串仍然不一定會被垃圾回收掉。也就是說即使駐留池中的字符串已經(jīng)沒有用處了,它可能也要等到CLR終結(jié)時(shí)才被銷毀。當(dāng)你使用Intern方法的時(shí)候,也應(yīng)該考慮到這個(gè)特殊的行為。
到此這篇關(guān)于.NET字符串駐留池的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
.Net?Core微服務(wù)rpc框架GRPC通信實(shí)際運(yùn)用
這篇文章介紹了.Net?Core微服務(wù)rpc框架GRPC通信實(shí)際運(yùn)用,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-01-01
.Net創(chuàng)建型設(shè)計(jì)模式之建造者、生成器模式(Builder)
這篇文章介紹了.Net設(shè)計(jì)模式之建造者、生成器模式(Builder),文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
.Net結(jié)構(gòu)型設(shè)計(jì)模式之橋接模式(Bridge)
這篇文章介紹了.Net結(jié)構(gòu)型設(shè)計(jì)模式之橋接模式(Bridge),文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
詳解Asp.net 5中的ApplicationBuilder
這篇文章介紹了Asp.net 5中的ApplicationBuilder,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-01-01
asp.net平臺下C#實(shí)現(xiàn)Socket通信
這篇文章介紹了asp.net平臺下C#實(shí)現(xiàn)Socket通信的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-01-01

