C# 反射(Reflection)

反射指程序可以访问、检测和修改它本身状态或行为的一种能力。

程序集包含模块,而模块包含类型,类型又包含成员。反射则提供了封装程序集、模块和类型的对象。

您可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型。然后,可以调用类型的方法或访问其字段和属性。

优缺点

优点:

  • 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: