NET面试题:C#中的using语句有什么作用

来源:互联网 发布:网络渗透测试教程 编辑:程序博客网 时间:2024/05/01 20:17

转自:http://www.izixue.com/thread-1191-1-1.html

NET面试题:C#中的using语句有什么作用

using可算是.NET中新的语法元素,它清楚地说明一个通常比较占用资源的对象何时开始使用和何时被手动释放。当using可以被使用时,笔者建议读者尽量使用using语句。至今为止,笔者发现它带给程序的只有优点,而没有任何弊端。
  所涉及的知识点
· using的基本语法
· using如何调用Dispose方法
  分析问题
在.NET的环境中,托管的资源都将由.NET的垃圾回收机制来释放,而一些非托管的资源则需要程序员手动地将它们释放。.NET提供了主动和被动两种释放非托管资源的方式,即IDisposalbe接口的Dispose方法和类型自己的Finalize方法,关于这两个方法和.NET的资源释放方式,笔者在本章的后续小节中会有详细的介绍,在本节读者需要了解的是:任何带有非托管资源的类型,都有必要实现IDisposalbe的Dispose方法,并且在使用完这些类型后需要手动地调用对象的Dispose方法来释放对象中的非托管资源。
如果类型正确地实现了Finalize方法,那即使Dispose方法不被调用,非托管资源也最终会被释放,但那时资源已经被很长时间无谓地占据了。
using语句的作用就是提供了一个高效的调用对象Dispose方法的方式。对于任何IDipose接口的类型,都可以使用using语句,而对于那些没有实现IDisposalbe接口的类型,使用using语句会导致一个编译错误。
先来看一下using语句的基本语法:
using(MemoryStream ms=new MemoryStream())
{
//做了很多工作
}
在上面代码中,using语句的一开始定义了一个MemoryStream的对象,之后在整个语句块中都可以使用ms,在using语句块结束的时候,ms的Dispose方法将会被自动调用。using语句不仅免除了程序员输入Dispose调用的代码,它还提供了机制保证Dispose方法被调用,无论using语句块顺利执行结束,还是抛出了一个异常。代码3-15演示了using的这一保护机制。
代码3-15  using:using.cs
namespace NET.MST.Third.Using
{
    public class Using
    {
        static void Main(string[] args)
        {
            try
            {
                //使用using
                using (MyDispose md = new MyDispose())
                {
                    md.DoWork();
                    //抛出一个异常来测试using
                    throw new Exception("得到一个异常");
                }
            }
            catch { }
            finally
            {
                Console.Read();
            }
        }
    }
    //仅仅用来作为测试,不使用任何非托管资源
    public class MyDispose : IDisposable
    {
        public void Dispose()
        {
            Console.WriteLine("Dispose方法被调用");
        }
        public void DoWork()
        {
            Console.WriteLine("做了很多工作");
        }
    }
}
在代码3-15中,using语句块抛出了一个异常,而该异常知道using语句结束后才被捕获。由于有了using语句的存在,即使异常被抛出,MyDispose的对象:md的Dispose方法仍然被调用了。下面是该程序的执行结果:
做了很多工作
Dispose方法被调用
事实上,C#编译器为using语句自动添加了try/finally块,所以Dispose方法能够保证被调用到,所以如下两段代码经过编译后内容将完全一致:
using (MyDispose md = new MyDispose())

    md.DoWork();
}

MyDispose md;
try
{
    md = new MyDispose();
    md.DoWork();
}
finally
{
    md.Dispose();
}
在彻底了解了using的实现原理后,读者还应该注意一点使用using时常犯的错误,那就是千万不要试图在using语句块外初始化对象,如下列代码所做的:
MyDispose md = new MyDispose();
using (md)
{
    md.DoWork();
}
看上去似乎没有任何问题,但是在多线程的程序中,上述代码就会有隐患。试想当md被初始化后程序突然产生一个异常而中断,那md对象中的非托管资源将没有机会得到释放,这对于系统来说危害是相当大的。所以读者在任何时候都应该在using语句中初始化需要使用的对象。
  答案
using语句为实现了IDisposable的类型对象调用Dipose方法,using语句能够保证使用的对象的Dispose方法在using语句块结束时被调用,无论是否有异常被抛出。C#编译器在编译时自动为using语句加上try/finally块,所以using的本质和异常捕捉语句一样,但语法更为简洁。所有using使用的对象都应该在using语句开始后再初始化,以保证所有的对象能够被Dispose。
原创粉丝点击