将不确定变为确定~程序是否真的Dispose了

简介:

首先将来说一下Dispose是什么东西吧,对于我们使用非托管的资源时,需要自己去实现Dispose这个方法,它的含义就是释放使用的内存空间。

例如Stream这个类型,它就是一个非托管类型,它会实现一个IDisposable接口,来实现Dispose方法

image

像TransactionScope,.net事务,它也是一个非托管的,也就是说,我们在使用完事务后,需要自己去进行Dispose()操作,下面问题就来了,这个Dispose写在哪里合适呢?

注意看这段代码:

using (TransactionScope trans = new TransactionScope())
         {
             try
             {
                 this.Update(order);
                 new WebAccountRecordsRepository().Insert(new WebAccountRecords
                 {
                     
                 });
                 new WebAccountBalancesRepository().Update(new WebAccountBalances { });
             }
             catch (Exception e)
             {
                // vm.AddItem(e.Message);
                 throw;
             }
             finally
             {
                 trans.Dispose(); 
             }
         }

这是非常标准的写到,完成一个订单处理的过程,它将处理订单,网站支付明细及网站总余额写在了一个事务里,这当然是没有问题的,注意看Dispose的位置,写在了finally{}里,这也是对的,当try{}完成后,将会执行finally片断,但注意catch{}段,它进行所有异常的捕捉,并进行抛出,好了,如果这个try{}段出现了异常,那finally{}段是否会执行吗?也就是dispose是否会被执行呢?

经过我的测试,它有执行,但由于你使用了throw,所以网页直接黄屏了,所以,最好把catch段进行处理,你可以去把异常写到日志里,但有一点要注意,finally{}块里不要写可能会出现异常的代码,否则,会使你的事务资源永远得到不释放!

例如:

using (TransactionScope trans = new TransactionScope())
       {
           try
           {
               this.Update(order);
               new WebAccountRecordsRepository().Insert(new WebAccountRecords
               {
                   
               });
               new WebAccountBalancesRepository().Update(new WebAccountBalances { });
           }
           catch (Exception e)
           {
              // vm.AddItem(e.Message);
               throw;
           }
           finally
           {
               Insert(list.First().id); //如果list集合为空,那这行会出现异常,导致它下面的代码将不能被执行
               trans.Dispose(); 
           }
       }

 当然,这一般是由于编程习惯引起的,大家以后注意就行了,在finally里释放资源时,应该考虑异常进行一些必要的判断。

如果非要写在finally里,如果你的对象不能确定是否会发生异常,那就try,catch吧,看代码:

finally

  {

       try {

            Insert(list.First().id); //如果list集合为空,那这行会出现异常,导致它下面的代码将不能被执行

           }

          catch (Exception e)

          {

            throw;

          }

         finally

          {

            trans.Dispose();

          }

}

这样,trans.Dispose()也是会被执行的,也就是说,对于同一个try,catch,finally来说,finally是永远都会被执行的。

本文转自博客园张占岭(仓储大叔)的博客,原文链接:将不确定变为确定~程序是否真的Dispose了,如需转载请自行联系原博主。

目录
相关文章
|
7月前
|
Linux API
Linux线程总结---线程的创建、退出、取消、回收、分离属性
Linux线程总结---线程的创建、退出、取消、回收、分离属性
|
小程序 C# 数据安全/隐私保护
C#小程序执行后及时清空控件中的数据
C#小程序执行后及时清空控件中的数据
132 0
C#小程序执行后及时清空控件中的数据
窗口置前的几种方法
函数功能:该函数改变一个子窗口,弹出式窗口式顶层窗口的尺寸,位置和Z序。子窗口,弹出式窗口,及顶层窗口根据它们在屏幕上出现的顺序排序、顶层窗口设置的级别最高,并且被设置为Z序的第一个窗口。
Qt窗口关闭和应用程序停止是否调用析构函数的一些说明
Qt窗口关闭和应用程序停止是否调用析构函数的一些说明
Qt窗口关闭和应用程序停止是否调用析构函数的一些说明
Thread类的sleep()方法和对象的wait()方法都可以让线程暂停执行,它们有什么区别?
sleep()方法(休眠)是线程类(Thread)的静态方法,调用此方法会让当前线程暂停执行指定的时间,将执行机会(CPU)让给其他线程,但是对象的锁依然保持,因此休眠时间结束后会自动恢复(线程回到就绪状态,请参考第66题中的线程状态转换图)。
1166 0