代码改变世界

4.2.1 使用C#的is和as操作符来转型

2011-12-08 10:21  iRead  阅读(325)  评论(0编辑  收藏  举报

  在C#语言中进行类型转换的另一种方式是使用is操作符。is检查一个对象是否兼容于指定的类型,并返回一个Boolean值:true或false。注意is操作符永远不会抛出异常,以下代码进行了演示:

  Object o = new Object();

  Boolean b1 = (o is Object);       //b1为true.

  Boolean b2 = (o is Employee);//b2为false.

  如果对象引用是null,is操作符总是返回false,因为没有可检查其类型的对象。is操作符通常像下面这样使用:

  if (o is Employee){

    Employee e = (Employee) o;

    //在if语句剩余的部分中使用e

  }

  在这段代码中,CLR实际会检查两次对象的类型。Is操作符首先核实o是否兼容于Employee类型。如果是,那么在if语句内部执行转型时,CLR再次核实o是否引用了一个Employee。CLR的类型检查增强了安全性,但无疑也会对性能造成一定影响。这是因为CLR首先必须判断变量(o)引用的对象的实际类型。然后,CLR必须遍历继承层次结构,用每个基类型去核对指定的类型(Employee)。由于这是一个相当常用的编程模式,所以C#专门提供了as操作符,目的就是简化这种代码的写法,同时提升其性能。

  Employee e = o as Employee;

  if (e != null){

    //在if语句中使用e

  }

  在这段代码中,CLR核实o是否兼容Employee类型;如果是,as会返回同一个对象的一个非null引用。如果o不兼容于Employee类型,as操作符会返回null。注意,as操作符造成CLR只校验一次对象的类型。if语句只是检查e是否为null。这个检查的速度比校验对象的类型快得多。

  as操作符的工作方式与强制类型转换一样,只是它永远不会抛出一个异常—相反,如果对象不能转型,结果就是null。所以,正确的做法是检查最终生成的引用是否为null。如果企图直接使用最终生成的引用,会抛出一个System.NullReferenceException异常。以下代码对此进行了演示:

  Object o = new Object();                             //新建一个Object对象

  Employee e = o as Employee;                    //将o转型为一个Employee

  //上述转型操作会失败,不会抛出异常,但e会被设为null

  e.ToString();                                                    //访问e会抛出一个NullReferenceException异常

   为了确定自己理解了上述内容,请完成以下小测验。假定以下两个类定义确实存在:

  internal class B{       //基类

  }

  internal class D : B{ //派生类

  }

  现在检查表4-3列出的C#代码。针对每一行代码,都用勾号注明该行代码是将成功编译和执行(OK),将造成编译时错误(CTE),还是将造成运行时错误(RTE)。

  表4-3 类型安全性测验

语句

OK

CTE

RTE

Object o1 = new Object();

 

 

Object o2 = new B();

 

 

Object o3 = new D();

 

 

Object o4=o3;

 

 

B b1 = new B();

 

 

B b2 = new D();

 

 

D d1 = new D();

 

 

B b3 = new Object();

 

 

D d2 = new Object();

 

 

B b4 = d1;

 

 

D d3 = b2;

 

 

D d4 = (D) d1;

 

 

D d5 = (D) b2;

 

 

D d6 = (D)b1;

 

 

B b5 = (B) o1;

 

 

B b6 = (D)b2;

 

 

 

注意:C#允许在一个类型中定义转换操作符方法,详情参见8.5节“转换操作符方法”。只有在使用一个转型表达式时,才会调用这些方法;使用C#的as或is操作符时,永远不会调用它们。