分享我在工作中遇到的多線程下導致RCW無法釋放的問題
最近有個同事在調(diào)用一個類庫中的方法時遇到了一個問題,異常信息如下:
嘗試釋放正在使用的RCW,活動線程或其他線程上正在使用該 RCW,釋放正在使用的 RCW 的嘗試會導致?lián)p壞或數(shù)據(jù)丟失。
該方法中對word文件進行相關(guān)了操作,因為我之前也在多線程環(huán)境下調(diào)用過該方法,并且沒遇到這個問題,所以同事讓我過去看看怎么回事。這個方法在對文件進行相關(guān)操作后,會調(diào)用另外一個方法釋放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對象并關(guān)閉word進程。因為異常信息定位在這里,所以我過去后就從這里開始看,但是看了半天,也沒看出類庫中的方法有什么問題,因為之前我使用的時候沒遇到這種情況,因此我覺得可能不會是這里的問題,并且我負責的那個產(chǎn)品已經(jīng)經(jīng)過了大量的測試,肯定是沒問題的,所以我說讓我看看你是怎么調(diào)用的吧,打開他的代碼看了一眼,整體上沒什么其他問題,但是有個地方引起了我的注意,代碼中對該類的實例化放在了全局范圍,因為是個cs項目,這么做會導致該對象始終被引用,因此即使在垃圾回收時也無法被釋放,而這里調(diào)用的又是com組件,就導致了word進程無法關(guān)閉,并且同事在這里用的是多線程,所以程序一運行起來,會出現(xiàn)一大堆word進程關(guān)不掉。于是就將這里的對象實例化放到了線程方法中,這樣在方法執(zhí)行結(jié)束后,堆中的對象就處于無引用狀態(tài),在垃圾回收時就被釋放了,問題就自然解決了。其實這里跟單線程還是多線程沒關(guān)系,主要是在全局范圍內(nèi)進行實例化導致了對象不能被垃圾回收,所以在寫代碼的時候一定要注意對象的生命周期。
相關(guān)文章
教你C#將CSV轉(zhuǎn)為Excel的實現(xiàn)方法
這篇文章主要介紹了C#?將CSV轉(zhuǎn)為Excel,轉(zhuǎn)換之后可執(zhí)行更多關(guān)于數(shù)據(jù)編輯、格式設(shè)置等操作,代碼簡單易懂,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2022-03-03C#通過經(jīng)緯度計算2個點之間距離的實現(xiàn)代碼
這篇文章主要介紹了C#通過經(jīng)緯度計算2個點之間距離實現(xiàn)代碼,本文對實現(xiàn)原理、經(jīng)緯度基本知識等一并做了講解,需要的朋友可以參考下2014-08-08