反射指程序可以访问、检测和修改它本身状态或行为的一种能力。
程序集包含模块,而模块包含类型,类型又包含成员。反射则提供了封装程序集、模块和类型的对象。
您可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型。然后,可以调用类型的方法或访问其字段和属性。
优缺点
优点:
- 1、反射提高了程序的灵活性和扩展性。
- 2、降低耦合性,提高自适应能力。
- 3、它允许程序创建和控制任何类的对象,无需提前硬编码目标类。
缺点:
- 1、性能问题:使用反射基本上是一种解释操作,用于字段和方法接入时要远慢于直接代码。因此反射机制主要应用在对灵活性和拓展性要求很高的系统框架上,普通程序不建议使用。
- 2、使用反射会模糊程序内部逻辑;程序员希望在源代码中看到程序的逻辑,反射却绕过了源代码的技术,因而会带来维护的问题,反射代码比相应的直接代码更复杂。
反射(Reflection)的用途
反射(Reflection)有下列用途:
- 它允许在运行时查看特性(attribute)信息。
- 它允许审查集合中的各种类型,以及实例化这些类型。
- 它允许延迟绑定的方法和属性(property)。
- 它允许在运行时创建新类型,然后使用这些类型执行一些任务。
查看元数据
我们已经在上面的章节中提到过,使用反射(Reflection)可以查看特性(attribute)信息。
System.Reflection 类的 MemberInfo 对象需要被初始化,用于发现与类相关的特性(attribute)。为了做到这点,您可以定义目标类的一个对象,如下:
System.Reflection.MemberInfo info = typeof(MyClass);
下面的程序演示了这点:
实例
using System ; [AttributeUsage (AttributeTargets . All ) ] public class HelpAttribute : System . Attribute { public readonly string Url ; public string Topic // Topic 是一个命名(named)参数 { get { return topic ; } set { topic = value ; } } public HelpAttribute ( string url ) // url 是一个定位(positional)参数 { this . Url = url ; } private string topic ; } [HelpAttribute ( "Information on the class MyClass" ) ] class MyClass { } namespace AttributeAppl { class Program { static void Main ( string [ ] args ) { System.Reflection . MemberInfo info = typeof (MyClass ) ; object [ ] attributes = info . GetCustomAttributes ( true ) ; for ( int i = 0 ; i < attributes . Length ; i ++ ) { System . Console . WriteLine (attributes [i ] ) ; } Console . ReadKey ( ) ; } } }
当上面的代码被编译和执行时,它会显示附加到类 MyClass 上的自定义特性:
HelpAttribute
实例
在本实例中,我们将使用在上一章中创建的 DeBugInfo 特性,并使用反射(Reflection)来读取 Rectangle 类中的元数据。
实例
using System ; using System.Reflection ; namespace BugFixApplication { // 一个自定义特性 BugFix 被赋给类及其成员 [AttributeUsage (AttributeTargets . Class | AttributeTargets . Constructor | AttributeTargets . Field | AttributeTargets . Method | AttributeTargets . Property, AllowMultiple = true ) ] public class DeBugInfo : System . Attribute { private int bugNo ; private string developer ; private string lastReview ; public string message ; public DeBugInfo ( int bg, string dev, string d ) { this . bugNo = bg ; this . developer = dev ; this . lastReview = d ; } public int BugNo { get { return bugNo ; } } public string Developer { get { return developer ; } } public string LastReview { get { return lastReview ; } } public string Message { get { return message ; } set { message = value ; } } } [DeBugInfo ( 45, "Zara Ali", "12/8/2012", Message = "Return type mismatch" ) ] [DeBugInfo ( 49, "Nuha Ali", "10/10/2012", Message = "Unused variable" ) ] class Rectangle { // 成员变量 protected double length ; protected double width ; public Rectangle ( double l, double w ) { length = l ; width = w ; } [DeBugInfo ( 55, "Zara Ali", "19/10/2012", Message = "Return type mismatch" ) ] public double GetArea ( ) { return length * width ; } [DeBugInfo ( 56, "Zara Ali", "19/10/2012" ) ] public void Display ( ) { Console . WriteLine ( "Length: {0}", length ) ; Console . WriteLine ( "Width: {0}", width ) ; Console . WriteLine ( "Area: {0}", GetArea ( ) ) ; } } //end class Rectangle class ExecuteRectangle { static void Main ( string [ ] args ) { Rectangle r = new Rectangle ( 4.5, 7.5 ) ; r . Display ( ) ; Type type = typeof (Rectangle ) ; // 遍历 Rectangle 类的特性 foreach ( Object attributes in type . GetCustomAttributes ( false ) ) { DeBugInfo dbi = (DeBugInfo )attributes ; if ( null != dbi ) { Console . WriteLine ( "Bug no: {0}", dbi . BugNo ) ; Console . WriteLine ( "Developer: {0}", dbi . Developer ) ; Console . WriteLine ( "Last Reviewed: {0}", dbi . LastReview ) ; Console . WriteLine ( "Remarks: {0}", dbi . Message ) ; } } // 遍历方法特性 foreach (MethodInfo m in type . GetMethods ( ) ) { foreach (Attribute a in m . GetCustomAttributes ( true ) ) { DeBugInfo dbi = (DeBugInfo )a ; if ( null != dbi ) { Console . WriteLine ( "Bug no: {0}, for Method: {1}", dbi . BugNo, m . Name ) ; Console . WriteLine ( "Developer: {0}", dbi . Developer ) ; Console . WriteLine ( "Last Reviewed: {0}", dbi . LastReview ) ; Console . WriteLine ( "Remarks: {0}", dbi . Message ) ; } } } Console . ReadLine ( ) ; } } }
当上面的代码被编译和执行时,它会产生下列结果:
Length: 4.5 Width: 7.5 Area: 33.75 Bug No: 49 Developer: Nuha Ali Last Reviewed: 10/10/2012 Remarks: Unused variable Bug No: 45 Developer: Zara Ali Last Reviewed: 12/8/2012 Remarks: Return type mismatch Bug No: 55, for Method: GetArea Developer: Zara Ali Last Reviewed: 19/10/2012 Remarks: Return type mismatch Bug No: 56, for Method: Display Developer: Zara Ali Last Reviewed: 19/10/2012 Remarks: