我正在尝试将C#与COM Interop库一起使用来打开一组非常繁重的Excel工作簿。我必须使用C#,因为我还需要启动宏,移动一些单元格,然后启动我公司使用的自定义excel-add-in。
然后我的程序退出,让工作簿保持打开状态,每个工作簿都在一个单独的excel实例中。我不希望在程序退出时关闭工作簿。
问题在于,当我的C#程序退出时,随着时间的推移,excel工作簿逐渐消耗更多内存,直到他们从原始的500 MB消耗3.5 GB的内存。
我曾经手工打开工作簿,而这些工作表从未消耗过那么多内存。一旦我开始使用C#打开它们,它们就会因极端内存使用而开始中断。我的理论是,不知何故,当我与COM Excel对象交互时,我创建了一个内存泄漏。
以下是我的原始代码:
using Excel = Microsoft.Office.Interop.Excel;
...
excelApp = new Excel.Application();
excelApp.Visible = true;
excelApp.Workbooks.Open(filename, misValue, misValue, misValue, misValue, misValue,
true, misValue, misValue, misValue, misValue, misValue, misValue, misValue, misValue);
excelApp.Calculation = Excel.XlCalculation.xlCalculationAutomatic;
我读到了你需要如何使用Marshal来释放用途,所以我现在正在尝试以下代码,但没有简单的方法来测试它,除了打开所有工作表并查看它们是否消耗了太多数据。
excelApp = new Excel.Application();
excelApp.Visible = true;
Excel.Workbooks currWorkbooks = excelApp.Workbooks;
Excel.Workbook currWorkbook = currWorkbooks.Open(filename, misValue, misValue, misValue, misValue, misValue,
true, misValue, misValue, misValue, misValue, misValue, misValue, misValue, misValue);
//excelApp.Calculation = Excel.XlCalculation.xlCalculationAutomatic;
int x = Marshal.ReleaseComObject(currWorkbook);
currWorkbook = null;
int y = Marshal.ReleaseComObject(currWorkbooks);
currWorkbooks = null;
答案 0 :(得分:16)
使用MS Office COM Interop库时,我遇到了一些避免内存泄漏的事情:
首先,“不要使用两个点”是记住它的最佳方式,但基本上,总是为新变量分配一个新的COM对象引用,不要链接成员,即使Intellisense鼓励它。链式调用会在后台执行一些阻止.NET框架正确发布的内容。以下是我用于启动Excel报告的一些代码:
//use vars for every COM object so references don't get leftover
//main Excel app
var excelApp = new Application();
var workbooks = excelApp.Workbooks;
//workbook template
var wbReport = workbooks.Add(@"C:\MyTemplate.xltx");
//Sheets objects for workbook
var wSheetsReport = wbReport.Sheets;
var wsReport = (Worksheet)wSheetsReport.get_Item("Sheet1");
其次,为创建的相反顺序创建的每个变量调用Marshal.ReleaseComObject()
,并在执行此操作之前调用几个垃圾收集方法:
//garbage collector
GC.Collect();
GC.WaitForPendingFinalizers();
//cleanup
Marshal.ReleaseComObject(wsReport);
Marshal.ReleaseComObject(wSheetsReport);
Marshal.ReleaseComObject(wbReport);
Marshal.ReleaseComObject(workbooks);
Marshal.ReleaseComObject(excelApp);
每次使用Excel时都使用这个方案已经解决了我的记忆问题,虽然它很乏味和悲伤,但我们不能像我们习惯的那样使用链式成员。