版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡介
第C#字符串內(nèi)存駐留機(jī)制分析在這之前我寫過一些文章來介紹關(guān)于字符串內(nèi)存分配和駐留的文章,涉及到的觀點(diǎn)主要有:字符串的駐留機(jī)制避免了對具有相同字符序列的字符串對象的重復(fù)創(chuàng)建;被駐留的字符串是不受GC管轄的,即被駐留的字符串對象不能被GC回收;被駐留的字符串是被同一進(jìn)程中所有應(yīng)用程序域共享的。至于具體的原因,相信在《關(guān)于CLR內(nèi)存管理一些深層次的討論》中,你可以找到答案。由于這些天來在做一些關(guān)于內(nèi)存泄露審查的工作,所以想通過具體的MemoryProfiling工具來為你證實(shí)上面的結(jié)論。我采用的MemoryProfiling工具是RedGate的ANTSMemoryProfiler,陷于篇幅問題我不對該工具進(jìn)行詳細(xì)的介紹,有興趣的朋友可以登錄它的官網(wǎng)。
一、具有相同字符序列的String對象不會(huì)重復(fù)創(chuàng)建
首先來證明第一個(gè)結(jié)論:具有相同字符序列的String對象不會(huì)重復(fù)創(chuàng)建。我先創(chuàng)建了一個(gè)簡單的Console應(yīng)用,編寫了如下的程序:在靜態(tài)方法BuildString中進(jìn)行了四次String對象的創(chuàng)建,str1和str2,str3和str4具有相同的值。該方法在Main方法中被執(zhí)行,在執(zhí)行前后通過調(diào)用Console.ReadLine方法讓程序Block住。
classProgram
staticvoidMain(string[]args)
Console.WriteLine("Pressanykeytobeginbuildingstring...");
Console.ReadLine();
BuildString();
Console.WriteLine("Pressanykeytoexit...");
Console.ReadLine();
staticvoidBuildString()
varstr1="ABCDEFG";
varstr2="ABCDEFG";
varstr3="1234678";
varstr4="1234678";
}
現(xiàn)在我們通過ANTSMemoryProfiler啟動(dòng)代碼這個(gè)Console程序的exe文件,在靜態(tài)方法前后(也就是相應(yīng)的文字被輸出到控制臺(tái)的時(shí)候)拍攝兩個(gè)內(nèi)存快照。通過比較這兩個(gè)快照下對象的變化,我們發(fā)現(xiàn)多了3個(gè)String類型的實(shí)例。
圖1
我們進(jìn)一步追蹤著多出的3個(gè)字符串的值到底是多少,于是我們查看實(shí)例列表。從下面的截圖中我們可以清晰地看到:除了一個(gè)值為byteIndex的字符串之外,另兩個(gè)的值分別為ABCDEFG和12345678,它們就是我們在靜態(tài)方法BuildString創(chuàng)建的。在BuildString方法中,我們創(chuàng)建了4個(gè)String對象,而在這里我們我們只看到了兩個(gè)。這無疑證實(shí)了字符串駐留機(jī)制的存在。
圖2
二、字符串駐留機(jī)制同樣于stringliteral+stringliteral的運(yùn)算
+是我們最為常見的字符串操作符,當(dāng)我們通過該操作符對兩個(gè)字符串進(jìn)行連接操作的時(shí)候,字符串的駐留機(jī)制依然有效。為此,我將BuildString方式定義成如下的方式,采用相同的Profiling流程,你依然可以看到與圖2完全一樣的結(jié)果。
staticvoidBuildString()
varstr1="ABCDEFG";
varstr2="ABCD"+"EFG";
varstr3="1234678";
varstr4="1234"+"678";
}
三、字符串駐留機(jī)智不適合Variable+stringliteral形式
雖然字符串的駐留適用于兩個(gè)通過引號(hào)括起來的字符串值直接進(jìn)行相加,但是如果將任何一個(gè)或者兩個(gè)換成字符串變量,最終運(yùn)算的結(jié)果是不能被駐留的。我們同樣可以通過類似于上面的步驟來證實(shí)這一點(diǎn),為此我們BuildString方法進(jìn)行了如下的修改。采用上面的Profiling流程,你看到的依然是圖2完全一樣的結(jié)果,也就是說無論是變量和一個(gè)字符串常量相加,還是兩個(gè)字符串常量相加,運(yùn)算的結(jié)果ABCDEFG1234678并沒有被駐留下來(實(shí)際上此時(shí)它已經(jīng)是一個(gè)垃圾對象,GC可以對其進(jìn)行回收)。
staticvoidBuildString()
varstr1="ABCDEFG";
varstr2="1234678";
varstr3="ABCDEFG"+str2;
varstr4=str1+"1234678";
varstr5=str1+str2;
}
四、調(diào)用string.Intern可以對運(yùn)算結(jié)果進(jìn)行強(qiáng)制駐留
雖然涉及到變量的字符串連接運(yùn)算結(jié)果不會(huì)被駐留,但是我們可以通過調(diào)用string.Intern方法對其進(jìn)行強(qiáng)制駐留,該方法會(huì)迫使傳入傳入?yún)?shù)表示的字符串被保存到駐留池中。為此,我們對BuildString方法進(jìn)行如下的修改:將ABCDEFG+str2運(yùn)算的結(jié)構(gòu)傳入string.Intern靜態(tài)方法中。
staticvoidBuildString()
varstr1="ABCDEFG";
varstr2="1234678";
varstr3=string.Intern("ABCDEFG"+str2);
}
通過采用上面的Profiling流程,在新創(chuàng)建對象(NewObject)String實(shí)例列表中,多出了一個(gè)ABCDEFG1234678。
圖3
五、駐留的字符串不能被GC回收
雖然String是一個(gè)引用類型,但是它卻不受GC管轄。GC在進(jìn)行回收的時(shí)候,看似垃圾對象的字符串實(shí)例依然保存在內(nèi)存中。為了演示,我們將BuildString方法還原成原來的代碼,并在調(diào)用該方法之后調(diào)用GC.Collect方法進(jìn)行強(qiáng)制垃圾回收。采用上面的Profiling流程,你看到的依然是圖2完全一樣的結(jié)果,四個(gè)本應(yīng)該是垃圾對象(str1~str4)在GC回收之后依然存在。
classProgram
staticvoidMain(string[]args)
Console.WriteLine("Pressanykeytobeginbuildingstring...");
Console.ReadLine();
BuildString();
GC.Collect();
Console.WriteLine("Pressanykeytoexit...");
Console.ReadLine();
staticvoidBuildString()
varstr1="ABCDEFG";
varstr2="ABCDEFG";
varstr3="1234678";
varstr4="1234678";
}
六、字符串駐留是基于整個(gè)進(jìn)程的
現(xiàn)在來證明最后一個(gè)結(jié)論:駐留的字符串是基于整個(gè)進(jìn)程范圍的,而不是基于當(dāng)前AppDomain。為了證明這個(gè)結(jié)論,我們可以要寫多一點(diǎn)代碼。我們借用《關(guān)于CLR內(nèi)存管理一些深層次的討論》中的方式,創(chuàng)建了如下一個(gè)AppDomainContext類,該類是對一個(gè)AppDomain對象的封裝。Invoke方法實(shí)現(xiàn)了在一個(gè)單獨(dú)的AppDomain中執(zhí)行某個(gè)基于泛型類型實(shí)例的操作。
publicclassAppDomainContext
publicAppDomainAppDomain{get;privateset;}
privateAppDomainContext(stringfriendlyName)
this.AppDomain=AppDomain.CreateDomain(friendlyName);
publicstaticAppDomainContextCreateDomainContext(stringfriendlyName)
returnnewAppDomainContext(friendlyName);
publicvoidInvokeT(ActionTaction)
Tinstance=(T)this.AppDomain.CreateInstanceAndUnwrap(typeof(T).Assembly.FullName,typeof(T).FullName);
action(instance);
}
然后我們將上述的BuildString方法實(shí)現(xiàn)在一個(gè)繼承自MarshalByRefObject的Foo類型中。
publicclassFoo:MarshalByRefObject
publicvoidBuildString()
varstr1="ABCDEFG";
varstr2="ABCDEFG";
varstr3="1234678";
varstr4="1234678";
}
然后再M(fèi)ain方法中,我們執(zhí)行如下的程序。下面的程序模擬的是創(chuàng)建了3個(gè)AppDomain,并在它們內(nèi)部進(jìn)行BuildString方法的執(zhí)行。如果字符串的駐留是基于AppDomain的話,應(yīng)該有6個(gè)String實(shí)例存在。但是采用上面的Profiling流程,你看到的依然圖2完全一樣的結(jié)果,這就充分證明了駐留機(jī)制是基于進(jìn)程而非AppDomain的結(jié)論。
staticvoidMain(string[]args)
Console.WriteLine("Pressanykeytobeginbuildingstring...");
Console.ReadLine();
AppDomainContext.CreateDomainContext("DomainA").InvokeFoo(foo=foo.BuildString());
AppDomainContext.CreateDomainContext("DomainB").InvokeFoo(foo=foo.BuildString());
AppD
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 小區(qū)消防安全評估指南
- 安全生產(chǎn)典范企業(yè)講解
- 2025-2026人教版小學(xué)二年級(jí)語文期末測試卷上
- 結(jié)構(gòu)專業(yè)考試題及答案
- 2025-2026人教版三年級(jí)語文上學(xué)期卷
- 腸道菌群與NAFLD肝硬化PHG:MDT調(diào)節(jié)策略
- 2025-2026一年級(jí)語文上學(xué)期期末測試卷
- 腸狹窄術(shù)后腹腔感染的處理策略
- 腸梗阻合并糖尿病患者的血糖管理策略
- 衛(wèi)生院執(zhí)業(yè)監(jiān)督管理制度
- 安全生產(chǎn)目標(biāo)及考核制度
- (2026版)患者十大安全目標(biāo)(2篇)
- 2026年北大拉丁語標(biāo)準(zhǔn)考試試題
- 臨床護(hù)理操作流程禮儀規(guī)范
- 2025年酒店總經(jīng)理年度工作總結(jié)暨戰(zhàn)略規(guī)劃
- 空氣栓塞課件教學(xué)
- 2025年國家市場監(jiān)管總局公開遴選公務(wù)員面試題及答案
- 肌骨康復(fù)腰椎課件
- 患者身份識(shí)別管理標(biāo)準(zhǔn)
- 2025年10月自考04184線性代數(shù)經(jīng)管類試題及答案含評分參考
- 2025年勞動(dòng)保障協(xié)理員三級(jí)技能試題及答案
評論
0/150
提交評論