在c#中我们可能经常看到using System,看多了就麻木了,可能有些朋友不知道using的其他的用法。这里我帮大家总结一下using的各个用法。
首先打开MSDN帮助,查找using可以找到解释如下:
using 关键字有两个主要用途:
- 作为指令,用于为命名空间创建别名或导入其他命名空间中定义的类型。请参见 using 指令。
- 作为语句,用于定义一个范围,在此范围的末尾将释放对象。请参见 using 语句。
好了我们看到了MSDN给了我们答案,其中using只要有两种用法一种是指令、另一种是语句。
那我们从指令开始吧,而在MSDN中指令又有两种用法
1.允许在命名空间中使用类型,这样,您就不必在该命名空间中限定某个类型的使用:
using System.Text;
2为命名空间或类型创建别名。
using Project = PC.MyCompany.Project;
(1.1)第一种我们在熟悉不过了,但是你知道vs的编译过程吗。当我们创建一个控制台应用程序,系统为我们生成了很多个默认的using空间。
例如:
1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Text;
但是当我们写一个hello world 程序的时候,真正用到的只有using System这个空间。可能你会说那我们引用这么多空间干嘛,不会占用资源吗?这里我来告诉你答案。虽然我们引用了很多多余的空间但是我们的vs却帮我们在内存中引用了它需要的那些。当然从我们角度看,有没有办法去掉那些我们不需要的命名空间呢?这里我告诉你一种办法。当我们做好整个项目的时候我们在using的区域内右击-----组织using------移除和排序,这样就可以去掉那些不需要的了。
(1.2)为命名空间或类型创建别名。这个用到的不是很多,但是讲到了那我们就详细点。当我们在一个命名空间内调用一个类的时候,没有有问题的。但是当我们调用在两个不同的命名空间内的相同名字的类名时,这时你是否想到办法呢。这就是别名的用处了,我们给两个空间分别取个别名,然后别名.类名,这样不是分开他们了吗。
1 // cs_using_directive2.cs 2 // Using directive. 3 using System; 4 // Using alias for a class. 5 using AliasToMyClass = NameSpace1.MyClass; 6 7 namespace NameSpace1 8 { 9 public class MyClass 10 { 11 public override string ToString() 12 { 13 return "You are in NameSpace1.MyClass"; 14 } 15 } 16 } 17 18 namespace NameSpace2 19 { 20 class MyClass 21 { 22 } 23 } 24 25 namespace NameSpace3 26 { 27 // Using directive: 28 using NameSpace1; 29 // Using directive: 30 using NameSpace2; 31 32 class MainClass 33 { 34 static void Main() 35 { 36 AliasToMyClass somevar = new AliasToMyClass(); 37 Console.WriteLine(somevar); 38 } 39 } 40 } 41 42
运行结果:
You are in NameSpace1.MyClass
(2)using (Font font3 = new Font("Arial", 10.0f),font4 = new Font("Arial", 10.0f))
首先Font 是访问非托管资源(本例中为文件句柄和设备上下文)的托管类型的示例。有许多其他类别的非托管资源和封装这些资源的类库类型。所有这些类型都必须实现 IDisposable 接口。
按照规则,当使用 IDisposable 对象时,应在 using 语句中声明和实例化此对象。using 语句按照正确的方式调用对象上的 Dispose 方法,并(在您按照前面所示方式使用它时)会导致在调用 Dispose 时对象自身处于范围之外。在 using 块中,对象是只读的并且无法进行修改或重新分配。
using 语句确保调用 Dispose,即使在调用对象上的方法时发生异常也是如此。通过将对象放入 try 块中,并在调用 finally 块中的 Dispose,可以获得相同的结果;实际上,这就是编译器转换 using 语句的方式。前面的代码示例在编译时将扩展到以下代码(请注意,使用额外的大括号为对象创建有限范围):
这句你理解了吗?那我们从IL代码看看using语句的实质。
1 .method private hidebysig static void Main(string[] args) cil managed 2 { 3 .entrypoint 4 // 代码大小 80 (0x50) 5 .maxstack 3 6 .locals init ([0] class [System.Drawing]System.Drawing.Font font3, 7 [1] class [System.Drawing]System.Drawing.Font font4, 8 [2] bool CS$4$0000) 9 IL_0000: nop 10 IL_0001: ldstr "Arial" 11 IL_0006: ldc.r4 10. 12 IL_000b: newobj instance void [System.Drawing]System.Drawing.Font::.ctor(string, 13 float32) 14 IL_0010: stloc.0 15 .try 16 { 17 IL_0011: ldstr "Arial" 18 IL_0016: ldc.r4 10. 19 IL_001b: newobj instance void [System.Drawing]System.Drawing.Font::.ctor(string, 20 float32) 21 IL_0020: stloc.1 22 .try 23 { 24 IL_0021: nop 25 IL_0022: nop 26 IL_0023: leave.s IL_0035 27 } // end .try 28 finally 29 { 30 IL_0025: ldloc.1 31 IL_0026: ldnull 32 IL_0027: ceq 33 IL_0029: stloc.2 34 IL_002a: ldloc.2 35 IL_002b: brtrue.s IL_0034 36 IL_002d: ldloc.1 37 IL_002e: callvirt instance void [mscorlib]System.IDisposable::Dispose() 38 IL_0033: nop 39 IL_0034: endfinally 40 } // end handler 41 IL_0035: nop 42 IL_0036: leave.s IL_0048 43 } // end .try 44 finally 45 { 46 IL_0038: ldloc.0 47 IL_0039: ldnull 48 IL_003a: ceq 49 IL_003c: stloc.2 50 IL_003d: ldloc.2 51 IL_003e: brtrue.s IL_0047 52 IL_0040: ldloc.0 53 IL_0041: callvirt instance void [mscorlib]System.IDisposable::Dispose() 54 IL_0046: nop 55 IL_0047: endfinally 56 } // end handler 57 IL_0048: nop 58 IL_0049: call valuetype [mscorlib]System.ConsoleKeyInfo [mscorlib]System.Console::ReadKey() 59 IL_004e: pop 60 IL_004f: ret 61 } // end of method Program::Main
可以看到其实using对应的是try、finally来处理。
所有上面的句子等同于
1 Font font3 = new Font("Arial", 10.0f); 2 try 3 { 4 Font font4 = new Font("Arial", 10.0f); 5 try 6 { 7 } 8 finally 9 { 10 if (font4 != null) 11 ((IDisposable)font4).Dispose(); 12 } 13 } 14 finally 15 { 16 if (font3 != null) 17 ((IDisposable)font3).Dispose(); 18 }
虽然using语句使用很简单,但是还是需要有几点注意的地方。
- Dispose方法用于清理对象封装的非托管资源,而不是释放对象的内存,对象的内存永远有垃圾回收器控制。
- 程序在达到using语句末尾时退出using块,而如果到达语句末尾之前引入异常则有可能提前退出。
- Using中初始化的对象,可以在using语句之前声明。
如:
Font font3 = new Font("Arial", 10.0f); Using(font3){ }
4.要使用的类必须实现IDisposable接口。