专栏首页程序员的SOD蜜彻底关闭Excle进程的几个方法

彻底关闭Excle进程的几个方法

之前研究过的问题,最近有朋友问,这里再总结下做一个笔记。

我们在应用程序里面通过创建Excle应用对象打开Excle的情况下,如果不注意几个问题,可能无法彻底关闭Excle进程,来考察下面的几种情况:

        public static void startexcel()
        {
            var  excel = new Microsoft.Office.Interop.Excel.Application();
            excel.Visible = true;
            var book=  excel.Application.Workbooks.Open("D:\\Book1.xlsx");
        }

上面的代码打开了一个工作簿,Excel启动了一个独立进程并且呈现界面给用户,不会再犯方法结束后关闭Excel。这种情况下本意是为了让用户决定何时关闭工作簿。

结果,当用户手工关闭工作簿后,Excle进程没有关闭,这是因为我们的.NET 托管代码打开的Excle的非托管代码,.NET运行时没有释放相关的句柄,需要加上下面几行代码来释放:

        public static void startexcel()
        {
            var  excel = new Microsoft.Office.Interop.Excel.Application();
            excel.Visible = true;
            var book=  excel.Application.Workbooks.Open("D:\\Book1.xlsx");

            System.Runtime.InteropServices.Marshal.ReleaseComObject(excel);
            excel = null;
            GC.Collect();
        }

上面的代码,Marshal.ReleaseComObject 会释放COM组件对象,这里是excel,然后,代码设置 excel=null,这样紧接着执行垃圾回收才有效,否则,无法回收excel句柄。

注意,执行上面的代码并不会关闭了Excel进程,它只是释放了Excle进程句柄与.NET运行时的关系。

当用户在外面手工关闭Excle窗体后,Excle进程才会真正从任务管理器消失。

有朋友可能说,我没有加上面三行代码也能够关闭Excle进程啊。

没错,上面的代码只不过是立即释放了Excle这种非托管资源。注意到我们的 excle对象是一个局部对象,所以当方法结束后,excle对象已经在方法堆栈上被清空了,只需要在外面合适的时候调用下垃圾回收,即可实现彻底关闭Excle进程的效果:

startexcel();
GC.Collect();
Console.WriteLine("excel close ok.");

如果我们的Excel进程不是由用户关闭而是要程序自动关闭怎么办?

这个时候只需要调用Excle应用程序对象的关闭方法即可。

完整的代码如下,并且下面的代码演示了Excle进程打开一个宏文件,然后再打开工作簿,处理事件,最后关闭Excle窗体,关闭进程清理资源的功能。

Excle的工作簿保存和关闭事件有时候比较有用,比如保存工作簿的时候就上传一份工作簿副本到服务器。

 public static void startexcel()
        {
            var  excel = new Microsoft.Office.Interop.Excel.Application();
            excel.Visible = true;
            excel.Workbooks.Open("C:\\A1000.xla");
            var book=  excel.Application.Workbooks.Open("D:\\Book1.xlsx");
            excel.WorkbookBeforeSave += Excel_WorkbookBeforeSave;
            excel.WorkbookBeforeClose += Excel_WorkbookBeforeClose;
            book.Close();
            excel.Quit();
            System.Runtime.InteropServices.Marshal.ReleaseComObject(excel);
            excel = null;
            GC.Collect();
        }

        private static void Excel_WorkbookBeforeClose(Workbook Wb, ref bool Cancel)
        {
            Console.WriteLine("Excel 关闭,title:" + Wb.Title);
        }

        private static void Excel_WorkbookBeforeSave(Workbook Wb, bool SaveAsUI, ref bool Cancel)
        {
            Console.WriteLine("Excel保存,title:"+Wb.Title);
        }

注:

本文的做法,也适用于关闭Word等其它Office程序。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 常见.NET功能代码汇总 (3) 33,彻底关闭Excel进程

    33,彻底关闭Excel进程 .NET中使用Excel属于使用非托管资源,使用完成后一般都要用GC回收资源,但是,调用GC的位置不正确,Excel进程可能无法彻...

    用户1177503
  • LJMM平台( Linux +Jexus+MySQL+mono) 上使用MySQL的简单总结

    近准备把PDF.NET框架的开源项目“超市管理系统”移植到Linux上跑(演示地址:http://221.123.142.196),使用Jexus服务器和MyS...

    用户1177503
  • PostgreSQL的.NET驱动程序Npgsql中参数对象的一个Bug

    最近将公司的项目从SqlServer移植到PostgreSQL数据库上来,在调用数据库的存储过程(自定义函数)的时候,发现一个奇怪的问题,老是报函数无法找到。 ...

    用户1177503
  • 16个好用的Excel小技巧合辑

    如果工作表的A列怎么都无法取消隐藏,肯定是窗格冻结了。视图 - 冻结窗格 - 取消冻结窗格。

    华章科技
  • Excel 的10个神奇功能,你会用几个?

    导读:在很多人的眼里,Excel只是一个简单的表格工具,大不了可以进行求和等简单的计算,他们这样认为我不怪他们,因为他们根本不了解Excel的神奇作用。

    华章科技
  • 使用node读写Excel文件

    薛定喵君
  • Java学习---C3P0的基本操作

    1.简介 C3P0是一个开源的JDBC连接池,它实现了数据源和JNDI绑定,支持JDBC3规范和JDBC2的标准扩展。目前使用它的开源项目有Hibernat...

    曼路
  • 假装我是一个牛B的架构师(五)

    上周接到HR小姐姐电话约面试,我说周末可不可以,然后就约在周六去了。这次的面试官问的还比较细,我一向本着尊重考官,找机会就装B的原则,顺着话题就聊下去。

    物流IT圈
  • iPhone多次输入错误密码锁机后刷机恢复(原有内容会丢失)

    俺踏月色而来
  • 一对一直播开发,为什么还需要CDN的帮助

    一对一直播开发的服务器压力远低于直播系统,为什么在开发时还是需要用到CDN的帮助?对于这种疑问,很多一对一直播开发的运营商都有过,其实多些对CDN的了解就会明白...

    yunbaokeji柯基

扫码关注云+社区

领取腾讯云代金券