处置 Microsoft.Office.Interop.Word.Application

问题描述 投票:0回答:9

(该帖子的后续内容(尚未得到答复):https://stackoverflow.com/q/6197829/314661

使用以下代码

Application app = new Application();
_Document doc = app.Documents.Open("myDocPath.docx", false, false, false);
doc.PrintOut(false);
doc.Close();

我正在尝试以编程方式打开并打印文件。

问题是每次我运行上述代码时都会启动一个新的 WINWORD.exe 进程,显然这很快就会耗尽所有内存。

应用程序类似乎不包含 dispose/close 或类似的方法。

经过一番研究,我(意识到)并将代码更改为以下内容。

 Application app = new Application();
 _Document doc = app.Documents.Open(fullFilePath + ".doc", false, false, false);
 doc.PrintOut(false);
 doc.Close();
 int res = System.Runtime.InteropServices.Marshal.ReleaseComObject(doc);
 int res1 = System.Runtime.InteropServices.Marshal.ReleaseComObject(app);

我可以看到剩余的引用计数为零,但进程仍然存在?

PS:我正在使用 Microsoft.Office.Interop 库的版本 14。

c# interop automation office-interop
9个回答
55
投票

不需要打电话

Quit
吗?

app.Quit();

46
投票

也许尝试设置

doc = null
并致电
GC.Collect()

编辑,不是我自己的代码,我忘记了从哪里得到它,但这是我用来处理Excel的,它完成了工作,也许你可以从中收集一些东西:

public void DisposeExcelInstance()
{
    app.DisplayAlerts = false;
    workBook.Close(null, null, null);
    app.Workbooks.Close();
    app.Quit();
    if (workSheet != null)
        System.Runtime.InteropServices.Marshal.ReleaseComObject(workSheet);
    if (workBook != null)
        System.Runtime.InteropServices.Marshal.ReleaseComObject(workBook);
    if (app != null)
        System.Runtime.InteropServices.Marshal.ReleaseComObject(app);
    workSheet = null;
    workBook = null;
    app = null;
    GC.Collect(); // force final cleanup!
}

6
投票

我认为似乎没有人注意到的主要问题是,如果 Word 已经打开,那么您不应该首先创建新的 Application 对象。 我们这些从 COM 和/或 VB6 时代就开始编码的人会记得 GetActiveObject。幸运的是.Net只需要一个ProgID。

推荐的操作方法如下:

try
{
    wordApp = (word.Application) Marshal.GetActiveObject("Word.Application");
}
catch(COMException ex) when (ex.HResult == -2147221021)
{
    wordApp = new word.Application();
}

5
投票

最好的解决方案..最后:

try {

    Microsoft.Office.Interop.Word.Application appWord = new Microsoft.Office.Interop.Word.Application();
    appWord.Visible = false;
    Microsoft.Office.Interop.Word.Document doc = null;
    wordDocument = appWord.Documents.Open((INP), ReadOnly: true);

    wordDocument.ExportAsFixedFormat(OUTP, Microsoft.Office.Interop.Word.WdExportFormat.wdExportFormatPDF);

    // doc.Close(false); // Close the Word Document.
    appWord.Quit(false); // Close Word Application.
} catch (Exception ex) {
    Console.WriteLine(ex.Message + "     " + ex.InnerException);
}

3
投票

您需要调用

app.Quit()
来关闭应用程序。我使用了下面的代码,它对我来说就像一个魅力 -

try
{
   Microsoft.Office.Interop.Word.Application wordApp = new Microsoft.Office.Interop.Word.Application();
   wordApp.Visible = false;
   Microsoft.Office.Interop.Word.Document doc = null;

   //Your code here...

   doc.Close(false); // Close the Word Document.
   wordApp.Quit(false); // Close Word Application.
}
catch (Exception ex)
{
   MessageBox.Show(ex.Message + "     " + ex.InnerException);
}
finally
{
   // Release all Interop objects.
   if (doc != null)
      System.Runtime.InteropServices.Marshal.ReleaseComObject(doc);
   if (wordApp != null)
      System.Runtime.InteropServices.Marshal.ReleaseComObject(wordApp);
   doc = null;
   wordApp = null;
   GC.Collect();
}

2
投票

同意其他海报,不需要

GC.Collect()
Marshal.ReleaseComObject()
。如果运行
app.Quit(false)
后该进程仍然存在,可能是因为您正在以不可见的方式运行应用程序,并且出现阻止应用程序关闭的提示,例如“文档恢复”对话框。如果是这种情况,您需要在创建应用程序时添加此内容。

app.DisplayAlerts = false;

1
投票

我关闭文档,然后关闭适合我的应用程序,然后强制垃圾回收。

// Document
object saveOptionsObject = saveDocument ? Word.WdSaveOptions.wdSaveChanges : Word.WdSaveOptions.wdDoNotSaveChanges;
this.WordDocument.Close(ref saveOptionsObject, ref Missing.Value, ref Missing.Value);

// Application
object saveOptionsObject = Word.WdSaveOptions.wdDoNotSaveChanges;
this.WordApplication.Quit(ref saveOptionsObject, ref Missing.Value, ref Missing.Value); 

GC.Collect();
GC.WaitForPendingFinalizers();

1
投票

试试这个..

doc.Close(false);
app.Quit(false);
if (doc != null)
    System.Runtime.InteropServices.Marshal.ReleaseComObject(doc);
if (app != null)
    System.Runtime.InteropServices.Marshal.ReleaseComObject(app);
doc = null;
app = null;
GC.Collect();

0
投票

就我而言,这是因为我打电话给

wordApp.Quit()
而不是
wordApp.Quit(false)
。刚刚添加
false
来调用此方法,现在可以正常工作了。

© www.soinside.com 2019 - 2024. All rights reserved.