国产日韩欧美一区二区三区三州_亚洲少妇熟女av_久久久久亚洲av国产精品_波多野结衣网站一区二区_亚洲欧美色片在线91_国产亚洲精品精品国产优播av_日本一区二区三区波多野结衣 _久久国产av不卡

?

.NET Framework GC.Keep Alive方法的實現(xiàn)原理分析

2021-08-07 11:56楊存?zhèn)?/span>
鄂州大學(xué)學(xué)報 2021年4期
關(guān)鍵詞:編譯器調(diào)用示例

楊存?zhèn)?/p>

(西南大學(xué) 計算機科學(xué)與技術(shù)系,重慶 402460)

在.NET中,內(nèi)存資源分為托管資源和非托管資源,其中托管資源指的是.NET 可以自動進行回收的資源,主要是指托管堆上分配的內(nèi)存資源[1]。GC 類中包含了垃圾回收相關(guān)的方法,其中GC.KeepAlive 是其中一個較為特別的方法,它利用編譯器和運行時的特性,阻止對象過早被回收。

1 GC.KeepAlive 的使用場景

示例1

在類Value中,實現(xiàn)了IDisposable中的Dispose方法,在類OuterClass 中,析構(gòu)方法調(diào)用類成員i 的Dispose 方法來釋放資源。

在Main 函數(shù)中,生成了一個Outerclass 的對象,變量名為outer,并將outer.i 作為參數(shù)傳遞給Do函數(shù)。outer.i 作為參數(shù)傳入之后,不存在對Outerclass 的對象的使用,因此垃圾回收器認為該對象已經(jīng)無用,而outer.i 是有效的。從這時起到程序運行結(jié)束的任何時刻,垃圾回收器都有可能執(zhí)行一次回收(回收Outerclass 的對象)。在垃圾回收器調(diào)用outer 的析構(gòu)方法后,outer.i 已經(jīng)執(zhí)行了Dispose 方法,從而Do 函數(shù)中對outer.i 的操作可能是無效的,造成程序出錯(圖1)。

圖1 不調(diào)用GC.KeepAlive 方法時程序的執(zhí)行流程

若在Do(outer.i)后添加“GC.KeepAlive(outer)”,則Do 方法調(diào)用之后依然有對outer 的使用,保證了在調(diào)用Do 方法的時候,OuterClass 的對象不會被垃圾回收器回收(圖2)[2]。

圖2 調(diào)用GC.KeepAlive 方法時程序的執(zhí)行流程

2 實現(xiàn)原理

Microsoft 對該方法的官方實現(xiàn)為:

2.1 屬性(Attribute)的使用

屬性(Attribute)是用于在運行時傳遞程序中各種元素(類、結(jié)構(gòu)體、方法等)的行為信息的聲明性質(zhì)的標簽,它添加元數(shù)據(jù),如編譯器指令和注釋、描述、方法、類等其他信息。

.NET 提供了兩種類型的特性:預(yù)定義特性和自定義特性。MethodImplAttribute 和Reliability-Contract 都屬于預(yù)定義特性[3]。

2.2 MethodImplAttribute 的使用

MethodImplAttribute 屬性指定了一個方法是怎樣編譯和執(zhí)行的。

ImplOptions.NoInlining 指定了該方法不應(yīng)該被內(nèi)聯(lián),由于GC.KeepAlive 并沒有對傳入的對象進行任何操作(也不應(yīng)該進行任何操作),假如將其內(nèi)聯(lián),將不能保證在調(diào)用該方法之前傳入的對象不被垃圾回收。

ReliabilityContract 屬性與實現(xiàn)原理無關(guān),與安全性和可靠性有關(guān),因此不做解析。

2.3 在調(diào)用GC.KeepAlive 之前對象不被回收的原因分析

.NET 的垃圾回收器使用一種引用跟蹤算法。在垃圾回收時,暫停所有線程,遍歷所有堆內(nèi)存中存在的對象,若引用類型變量引用了某個對象,該對象的同步索引字段將被置為1,否則為0。完成遍歷后,同步索引字段為0 的對象是可以被回收的(不一定會被回收)[4]。

在示例1 中,outer.i 作為參數(shù)傳入后,假設(shè)垃圾回收器開始回收,將發(fā)現(xiàn)OuterClass 的對象的同步索引塊標記為0,這意味著沒有變量繼續(xù)引用OuterClass 的對象,則其可以回收。

若加上“GC.KeepAlive(outer)”,則outer.i 作為參數(shù)傳入后,假如垃圾回收器開始回收,發(fā)現(xiàn)在Main 函數(shù)中還有待調(diào)用的GC.KeepAlive 方法使用了OuterClass 的對象,因此在調(diào)用GC.KeepAlive 之前,同步索引塊都為1,即該對象都必須存在,不應(yīng)該被回收。假設(shè)將“GC.KeepAlive(outer)”替換成“var str=outer.ToString ()” 等語句(只要使用了OuterClass 的對象),和調(diào)用GC.KeepAlive 方法的作用是一樣的,只是該方法不產(chǎn)生任何副作用。

3.NET 源碼中對該方法運用的示例分析

3.1 GC.GetGeneration 方法的實現(xiàn)

示例2:

.NET 的垃圾回收器查找未被引用的對象時,使用可達性分析算法[5],WeakReference 引用變量不會存在于GCRoots 開始的引用鏈中。若在調(diào)用GetGenerationWR 方法之后不調(diào)用KeepAlive 方法,不能保證wo 所指向的對象一定是存在的,造成調(diào)用失敗。

3.2 X509Utils.GetCertContext 方法的實現(xiàn)

示例3:

在將certificate.Handle 作為參數(shù)傳入CertDuplicateCertificateContext 方法后,若沒有GC.KeepAlive 方法,將導(dǎo)致certificate 隨時被垃圾回收,導(dǎo)致certificate.Handle 失效。

3.3 RuntimeHelpers.PrepareMethod 方法的實現(xiàn)

示例4:

PrepareMethod 方法是對外部函數(shù)_PrepareMethod 的一層包裝,在CopyRuntimeTypeHandles 方法使用instantiation 作為參數(shù)后,instantiationHandles 的有效性依然依賴于instantiation,故在_PrepareMethod 函數(shù)后調(diào)用GC.KeepAlive 方法來保證instantiation 在PrepareMethod 函數(shù)返回前都是有效的。

4 結(jié)語

GC.KeepAlive 的實現(xiàn)并未調(diào)用.NET 中的內(nèi)部類和方法,也未調(diào)用Win32 的API,其實質(zhì)是運用了編譯器和運行時的特性,保證了對象不被過早回收,其實現(xiàn)十分簡潔。在實際運用中準確運用此方法,提高了程序的安全性和穩(wěn)定性。

猜你喜歡
編譯器調(diào)用示例
基于相異編譯器的安全計算機平臺交叉編譯環(huán)境設(shè)計
2019年高考上海卷作文示例
核電項目物項調(diào)用管理的應(yīng)用研究
常見單位符號大小寫混淆示例
常見單位符號大小寫混淆示例
LabWindows/CVI下基于ActiveX技術(shù)的Excel調(diào)用
“全等三角形”錯解示例
基于系統(tǒng)調(diào)用的惡意軟件檢測技術(shù)研究
Microchip為MPLAB XC系列專業(yè)版編譯器推出低成本可續(xù)訂包月許可證
通用NC代碼編譯器的設(shè)計與實現(xiàn)
四子王旗| 汉川市| 宿州市| 武城县| 长治市| 克山县| 定州市| 醴陵市| 山丹县| 乐平市| 平南县| 仁寿县| 泰兴市| 安陆市| 垫江县| 江孜县| 五台县| 珲春市| 定远县| 鹤山市| 利津县| 阜南县| 体育| 丰顺县| 桓台县| 涞源县| 得荣县| 德清县| 庄浪县| 临朐县| 收藏| 鞍山市| 汉源县| 乳源| 祁门县| 从化市| 荔波县| 罗甸县| 美姑县| 贡觉县| 北碚区|