最近有個(gè)同事在調(diào)用一個(gè)類(lèi)庫(kù)中的方法時(shí)遇到了一個(gè)問(wèn)題,異常信息如下:
嘗試釋放正在使用的RCW,活動(dòng)線程或其他線程上正在使用該 RCW,釋放正在使用的 RCW 的嘗試會(huì)導(dǎo)致?lián)p壞或數(shù)據(jù)丟失。
該方法中對(duì)word文件進(jìn)行相關(guān)了操作,因?yàn)槲抑耙苍诙嗑€程環(huán)境下調(diào)用過(guò)該方法,并且沒(méi)遇到這個(gè)問(wèn)題,所以同事讓我過(guò)去看看怎么回事。這個(gè)方法在對(duì)文件進(jìn)行相關(guān)操作后,會(huì)調(diào)用另外一個(gè)方法釋放word對(duì)象,部分代碼如下:
Word._Application t = oWord as Word._Application;object oIsSave = false;t.Quit(ref oIsSave, ref oMissing, ref oMissing);System.Runtime.InteropServices.Marshal.ReleaseComObject(oDoc);System.Runtime.InteropServices.Marshal.ReleaseComObject(oWord);oWord = null;oDoc = null;GC.Collect();GC.Collect();
該段代碼是為了保證立即釋放word對(duì)象并關(guān)閉word進(jìn)程。因?yàn)楫惓P畔⒍ㄎ辉谶@里,所以我過(guò)去后就從這里開(kāi)始看,但是看了半天,也沒(méi)看出類(lèi)庫(kù)中的方法有什么問(wèn)題,因?yàn)橹拔沂褂玫臅r(shí)候沒(méi)遇到這種情況,因此我覺(jué)得可能不會(huì)是這里的問(wèn)題,并且我負(fù)責(zé)的那個(gè)產(chǎn)品已經(jīng)經(jīng)過(guò)了大量的測(cè)試,肯定是沒(méi)問(wèn)題的,所以我說(shuō)讓我看看你是怎么調(diào)用的吧,打開(kāi)他的代碼看了一眼,整體上沒(méi)什么其他問(wèn)題,但是有個(gè)地方引起了我的注意,代碼中對(duì)該類(lèi)的實(shí)例化放在了全局范圍,因?yàn)槭莻€(gè)cs項(xiàng)目,這么做會(huì)導(dǎo)致該對(duì)象始終被引用,因此即使在垃圾回收時(shí)也無(wú)法被釋放,而這里調(diào)用的又是com組件,就導(dǎo)致了word進(jìn)程無(wú)法關(guān)閉,并且同事在這里用的是多線程,所以程序一運(yùn)行起來(lái),會(huì)出現(xiàn)一大堆word進(jìn)程關(guān)不掉。于是就將這里的對(duì)象實(shí)例化放到了線程方法中,這樣在方法執(zhí)行結(jié)束后,堆中的對(duì)象就處于無(wú)引用狀態(tài),在垃圾回收時(shí)就被釋放了,問(wèn)題就自然解決了。其實(shí)這里跟單線程還是多線程沒(méi)關(guān)系,主要是在全局范圍內(nèi)進(jìn)行實(shí)例化導(dǎo)致了對(duì)象不能被垃圾回收,所以在寫(xiě)代碼的時(shí)候一定要注意對(duì)象的生命周期。
新聞熱點(diǎn)
疑難解答
圖片精選