夏虫不可语冰
此次事件讓我再次體會(huì)到了什么是“夏蟲(chóng)不可語(yǔ)冰”。博客園有些人壓根沒(méi)看完你的文章,帶著已有認(rèn)知斷章取義不暇思索上來(lái)就是噴。不過(guò),以前也遇到過(guò),現(xiàn)在也慢慢習(xí)慣了。
對(duì)于托管堆內(nèi)存泄漏的說(shuō)法,不管認(rèn)不認(rèn)同,我還是希望我的粉絲們能知道有這么個(gè)定義,有這么一回事。相信你們以后在閱讀文章或與人交流的時(shí)候,還會(huì)看到或聽(tīng)到有人管這種現(xiàn)象叫內(nèi)存泄漏,屆時(shí)也不至于蒙圈。雖然這是個(gè)很小的知識(shí)點(diǎn)。
C#中的匿名函數(shù)是頭等對(duì)象,可以作為變量使用,也可以作為參數(shù)傳遞。Task.Run 中使用匿名函數(shù)作為參數(shù)就是一個(gè)典型的應(yīng)用場(chǎng)景。如果匿名函數(shù)捕獲了所在類(lèi)的成員,對(duì)外部調(diào)用者來(lái)說(shuō),這個(gè)類(lèi)的實(shí)例什么時(shí)候被釋放是無(wú)法預(yù)知的(除非你十分清楚這個(gè)類(lèi)的內(nèi)部實(shí)現(xiàn)),但你知道它最終是會(huì)被回收的。有人把這種可能導(dǎo)致對(duì)象延遲回收的現(xiàn)象定義為內(nèi)存泄漏,也有很多人不認(rèn)為是內(nèi)存泄漏。這是定義問(wèn)題,如果你能用一個(gè)詞來(lái)更好地定義這種現(xiàn)象也是可以接受的。
Many share the opinion that?managed memory leaks?are not memory leaks at all since they are still referenced and theoretically can be de-allocated. It’s a matter of definition and my point of view is that they are indeed?memory leaks. They hold memory that can’t be allocated for another instance and will eventually cause an out-of-memory exception.
Ref: dwz.date/d48U
很早之前我就看過(guò)類(lèi)似的文章,把匿名方法捕獲類(lèi)成員導(dǎo)致延遲回收的現(xiàn)象定義為內(nèi)存泄漏。上面引用的這篇是最近閱讀到的一篇,這位外國(guó)作者寫(xiě)了很多 .NET 內(nèi)存管理相關(guān)的文章,都值得一讀。
如果只站在操作系統(tǒng)的角度看內(nèi)存是否被掌控來(lái)定義內(nèi)存泄漏,.NET 托管堆幾乎就沒(méi)有不被掌控的內(nèi)存占用(托管堆之所以叫“托管”就是 CRL 會(huì)自動(dòng)管理)。而之所以有人定義了托管內(nèi)存泄漏,是因?yàn)閷?duì) CLR 來(lái)說(shuō),在局部視角不再需要使用的對(duì)象依然被其它實(shí)例引用,可能導(dǎo)致 GC 恰巧在搜索 Root 的時(shí)候不能把這個(gè)對(duì)象所占內(nèi)存分配給其它實(shí)例。對(duì)于 CLR 來(lái)說(shuō),它對(duì)托管對(duì)象有全權(quán)“托管”之責(zé)。如果在被引用期間,恰巧 GC 在搜索 Root,就會(huì)導(dǎo)致 GC 錯(cuò)過(guò)了一次回收該對(duì)象的機(jī)會(huì),此次任務(wù)就“失職”了,只能在下一次搜索時(shí)回收它。但如果使用了本地(局部)變量,就可以避免這種情況。
大部分場(chǎng)景,我們并不需要在意這一點(diǎn)性能,何況這是個(gè)概率很低的事件。但是它值得引起注意,哪怕只有萬(wàn)分之一的概率。在一些特別的場(chǎng)景下(比如高并發(fā)),遇到這個(gè)問(wèn)題我們需要知道使用本地變量來(lái)優(yōu)化我們的程序。
這個(gè)現(xiàn)象是不是內(nèi)存泄漏,只是個(gè)定義問(wèn)題,如果覺(jué)得這個(gè)定義不妥,可以不認(rèn)同這個(gè)定義。正如 @楚人Le 說(shuō)的,你也可以把它定義為“空間泄漏”:
我傾向于認(rèn)為這屬于內(nèi)存泄漏。Wikipedia上關(guān)于Memory leak有這樣一句話:“A space leak occurs when a computer program uses more memory than necessary. In contrast to memory leaks, where the leaked memory is never released, the memory consumed by a space leak is released, but later than expected”。如果我們非常較真的話,或許可以使用“空間泄露”這個(gè)詞。當(dāng)計(jì)算機(jī)程序使用超過(guò)所需的內(nèi)存時(shí),就會(huì)發(fā)生空間泄漏。與內(nèi)存泄漏相比,空間泄漏所消耗的內(nèi)存會(huì)被釋放,但會(huì)比預(yù)期的要晚。
這只是個(gè)定義,這只是個(gè)定義,這只是個(gè)定義。不必糾結(jié),更沒(méi)必要因?yàn)椴徽J(rèn)同就惡語(yǔ)相向。
今天的文章夾雜著一點(diǎn)情緒,請(qǐng)大家見(jiàn)諒。要說(shuō)一點(diǎn)都不在意那些評(píng)論,那肯定是假的,心中多少還是會(huì)有些不愉悅的。面對(duì)網(wǎng)絡(luò)噴子,如果內(nèi)心不夠強(qiáng)大,還真是不敢隨便發(fā)表文章。這次也讓我意識(shí)到,以后有爭(zhēng)議性的文章少寫(xiě),大家也是一樣。
自媒體很多大V都經(jīng)歷過(guò)這種時(shí)段,和他們相比我這個(gè)蝦兵小卒遇到的這么三兩次不算什么。正如一線碼農(nóng)大佬說(shuō)的,有忠粉就有黑粉。既然走上這條路,必然也要經(jīng)歷這些,這便是成長(zhǎng)。
每每遇到這種事情,就特別想和鐵粉們說(shuō)聲:感謝!感謝你們一路以來(lái)的陪伴、理解、支持和鼓勵(lì),感謝你們?cè)敢夂臀乙黄饘W(xué)習(xí)和成長(zhǎng)。
總結(jié)
 
                            
                        - 上一篇: 微软用的工具,统一财务三大表及高级分析通
- 下一篇: 展望2021,Java、Go、.NET,
