Linq String類型淺析
昨天調(diào)試一段程序發(fā)現(xiàn)內(nèi)存始終釋放不掉,最后終于發(fā)現(xiàn)是對(duì)String 的錯(cuò)誤使用造成,這促使我今天又仔細(xì)研究了一下Linq String類型,不研究不知道,一研究發(fā)現(xiàn)我過(guò)去對(duì)String 的很多認(rèn)識(shí)都是錯(cuò)誤的,感覺(jué)這種錯(cuò)誤認(rèn)識(shí)還比較有典型性,于是寫(xiě)下此文和大家一起探討。
1. Linq String類型變量追加,或修改后的新String對(duì)象是駐留(Interned)的。
我過(guò)去想當(dāng)然的認(rèn)為s2 是駐留的,但實(shí)際上并非如此,用 string.IsInterned 方法檢測(cè)s2是非駐留的。后來(lái)研究發(fā)現(xiàn)只有常量字符串才會(huì)默認(rèn)駐留,其他的字符串變量哪怕是采用 new string 構(gòu)造出來(lái)的,默認(rèn)都非駐留,除非用string.Intern 強(qiáng)行駐留。后面我將提到駐留對(duì)內(nèi)存的影響,微軟之所以不讓所有的字符串都駐留,我認(rèn)為還是處于內(nèi)存方面的考慮。
2. String 變量不再引用后CLR會(huì)通過(guò)GC自動(dòng)釋放其內(nèi)存。
- string s1 = "abcd";
- s1 = null;
上面代碼,我想當(dāng)然的認(rèn)為s1 = null 后已經(jīng)不再對(duì) "abcd" 這個(gè)字符串引用,如果沒(méi)有其他引用指向這個(gè)字符串,GC會(huì)釋放"abcd"這塊內(nèi)存。實(shí)際結(jié)果卻是否定的。因?yàn)閟1 被賦予了一個(gè)常量,導(dǎo)致 "abcd"這個(gè)字符串是駐留的,駐留的字符串在進(jìn)程結(jié)束之前無(wú)法被自動(dòng)釋放。更糟糕的是,我昨天調(diào)試的那段程序里面大量的字符串變量被采用 string.Intern 強(qiáng)制駐留,這導(dǎo)致我把所有的托管對(duì)象都釋放了依然無(wú)法釋放那部分大概30多M的內(nèi)存。
遺憾的是微軟的MSDN中文版中string.Intern 的幫助信息里面竟然漏掉了性能考諒(Performance consideration) 這一節(jié),我估計(jì)大多數(shù)中國(guó)程序員包括我在內(nèi)如果有中文的幫助是懶得去看英文的。很遺憾微軟中文的幫助不知道為什么把最重要的部分給漏了。
3. 兩個(gè)String如果引用不同只能用Equal 比較。
我一直想當(dāng)然的認(rèn)為 兩個(gè)Linq String類型如果用 == 操作符比較,將比較其引用。所以如果兩個(gè)String引用不同,則只能使用Equal 來(lái)比較它們是否相等。
【編輯推薦】