分享我在工作中遇到的多線程下導致RCW無法釋放的問題
最近有個同事在調用一個類庫中的方法時遇到了一個問題,異常信息如下:
嘗試釋放正在使用的RCW,活動線程或其他線程上正在使用該 RCW,釋放正在使用的 RCW 的嘗試會導致?lián)p壞或數(shù)據(jù)丟失。
該方法中對word文件進行相關了操作,因為我之前也在多線程環(huán)境下調用過該方法,并且沒遇到這個問題,所以同事讓我過去看看怎么回事。這個方法在對文件進行相關操作后,會調用另外一個方法釋放word對象,部分代碼如下:
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對象并關閉word進程。因為異常信息定位在這里,所以我過去后就從這里開始看,但是看了半天,也沒看出類庫中的方法有什么問題,因為之前我使用的時候沒遇到這種情況,因此我覺得可能不會是這里的問題,并且我負責的那個產品已經經過了大量的測試,肯定是沒問題的,所以我說讓我看看你是怎么調用的吧,打開他的代碼看了一眼,整體上沒什么其他問題,但是有個地方引起了我的注意,代碼中對該類的實例化放在了全局范圍,因為是個cs項目,這么做會導致該對象始終被引用,因此即使在垃圾回收時也無法被釋放,而這里調用的又是com組件,就導致了word進程無法關閉,并且同事在這里用的是多線程,所以程序一運行起來,會出現(xiàn)一大堆word進程關不掉。于是就將這里的對象實例化放到了線程方法中,這樣在方法執(zhí)行結束后,堆中的對象就處于無引用狀態(tài),在垃圾回收時就被釋放了,問題就自然解決了。其實這里跟單線程還是多線程沒關系,主要是在全局范圍內進行實例化導致了對象不能被垃圾回收,所以在寫代碼的時候一定要注意對象的生命周期。