C#之CLR內(nèi)存字符串常量池(string)
C#中的string是比特殊的類,說引用類型,但不存在堆里面,而且String str=new String("HelloWorld")這樣的重裝也說沒有的。
我們先來看一個方法:
class Program { static void Main(string[] args) { String s = "HelloWorld"; Console.WriteLine(s); } }
然后我們用ildasm.exe工具把它生成IL語言來看一看它里面是怎么玩的:
.method private hidebysig static void Main(string[] args) cil managed { .entrypoint // Code size 15 (0xf) .maxstack 1 .locals init ([0] string s) IL_0000: nop IL_0001: ldstr "HelloWorld" IL_0006: stloc.0 IL_0007: ldloc.0 IL_0008: call void [mscorlib]System.Console::WriteLine(string) IL_000d: nop IL_000e: ret } // end of method Program::Main
我們在里面并沒有看見newObj(所以我們認為不在堆里面)的指令,只有一個特殊ldstr(load string)指令,它用從元數(shù)據(jù)獲取一個文本常量字符串(字符串常量池)構造一個String對象。這證明了CLR說用一種特殊的方式構造了字符串。
我們再舉一個簡單例子看看:
class Program { static void Main(string[] args) { String s = "HelloWorld"; s = "HelloC#"; s = "HelloJava"; String s1= "HelloC#"; Console.WriteLine(s); } }
對照這個例子我們來看看內(nèi)存圖是怎么走的:
首先CLR內(nèi)部機制會在運行這個方法之前就會有"prologue"代碼去開辟內(nèi)存空間,s和s1就說這個時候創(chuàng)建的。
我們創(chuàng)建了一個s的字符串對象,賦值為HelloWorld,把s插入棧,然后內(nèi)部機制去字符串常量池中找HelloWorld副本,發(fā)現(xiàn)沒有找到就會創(chuàng)建一個,接著會去保存這個HelloWorld在字符串常量池中的地址(Line1)。然后我們?yōu)閟對象在賦值為HelloC#,由于同一個對象,棧中不做操作,去字符串常量池中找,沒找到則創(chuàng)建,然后修改s所存儲的地址(line 2),HelloJava同樣的操作。 再創(chuàng)建一個s1的string對象,把s1壓入棧,為S1賦值HelloC#,這個時候會去字符常量池中找,找到了就存這個引用。
相關文章
c#根據(jù)網(wǎng)址抓取網(wǎng)頁截屏生成圖片的示例
本文主要介紹了c#根據(jù)網(wǎng)址抓取網(wǎng)頁截屏生成圖片并保存的示例,代碼中使用了WebBrowser控件來完成這個功能,大家參考使用吧2014-01-01