zoukankan      html  css  js  c++  java
  • 反射_01概述和反射中的运行时类型以及查看类型信息

    反射概述

    公共语言运行库加载器管理应用程序域,这些域在拥有相同应用程序范围的对象周围形成确定边界。这种管理包括将每个程序集加载到相应的应用程序域,以及控制每个程序集中类型层次结构的内存布局。
    程序集包含模块,而模块包含类型,类型又包含成员。反射则提供了封装程序集、模块和类型的对象。可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型。然后,调用类型的方法或访问其字段和属性。反射通常具有以下用途:
    1)        使用 Assembly 定义和加载程序集,加载程序集清单中列出的模块,以及从此程序集中查找类型,并创建该类型的实例。
    2)        使用 Module 发现以下信息:包含模块的程序集,以及模块中的类等。还可以获取在模块上定义的所有全局方法或其他特定的非全局方法。
    3)        使用 ConstructorInfo 发现以下信息:构造函数的名称、参数、访问修饰符(如 public 或 private)和实现详细信息(如 abstract 或 virtual)等。
    使用 Type 的 GetConstructors 或GetConstructor 方法来调用特定的构造函数。
    4)        使用 MethodInfo 发现以下信息:方法的名称、返回类型、参数、访问修饰符(如 public 或 private)和实现详细信息(如 abstract 或 virtual)等
    使用 Type 的 GetMethods 或GetMethod 方法来调用特定的方法。
    5)        使用 FieldInfo 发现以下信息:字段的名称、访问修饰符(如 public 或 private)和实现详细信息(如 static)等,并获取或设置字段值。
    6)        使用 EventInfo 发现以下信息:事件的名称、事件处理程序数据类型、自定义属性、声明类型和反射类型等,并添加或移除事件处理程序。
    7)        使用 PropertyInfo 发现以下信息:属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,并获取或设置属性值。
    8)        使用 ParameterInfo 发现以下信息:参数的名称、数据类型、参数是输入参数还是输出参数,以及参数在方法签名中的位置等。
    9)        当您在一个应用程序域的仅反射上下文中工作时,请使用 CustomAttributeData 来了解有关自定义属性的信息。使用 CustomAttributeData,您不必创建属性的实例就可以检查它们。
    System.Reflection.Emit 命名空间的类提供了一种特殊形式的反射,使您能够在运行时生成类型。
    反射也可用于创建称作类型浏览器的应用程序,使用户能够选择类型,然后查看有关选定类型的信息。
    反射还有其他一些用途。System.Runtime.Serialization 命名空间中的类使用反射来访问数据,并确定要持久保存的字段。System.Runtime.Remoting 命名空间中的类通过序列化来间接地使用反射。

    反射中的运行时类型

    反射提供类(例如 Type 和 MethodInfo)来表示类型、成员、参数和其他代码实体。但是,在您使用反射时,您并不直接使用这些类,这些类中的大多数是抽象的。 您使用的是公共语言运行时 (CLR) 提供的类型。
    例如,使用 C# 的 typeof 运算符。获取 Type 对象时,该对象实际上是 RuntimeType。 RuntimeType 派生自 Type,并提供所有抽象方法的实现。
    这些运行时类是 internal。 它们的文档与它们的基类的文档并没有分开,因为它们的行为是由基类文档描述的。

    查看类型信息

    System.Type 类对于反射起着核心的作用。 当反射请求加载的类型时,公共语言运行时将为它创建一个 Type。您可以使用 Type 对象的方法、字段、属性和嵌套类来查找有关该类型的所有信息。
    使用 Assembly.GetType 或 Assembly.GetTypes 从尚未加载的程序集中获取 Type 对象,并传入所需类型的名称。
    使用 Type.GetType 可从已加载的程序集中获取 Type 对象。
    使用Module.GetType 和 Module.GetTypes 可获取模块 Type 对象。
    注意:如果想要检查和操作泛型类型和方法,请参见反射类型和泛型类型和如何:使用反射检查和实例化泛型类型中提供的附加信息。
    下面的示例显示在获取程序集的 Assembly 对象和模块时所必需的语法。
    // 从 object 对象获得程序集mscorlib
    Assembly a = typeof(object).Module.Assembly;
    下面的示例说明如何从已加载的程序集中获取 Type 对象。

       1: using System;
       2: using System.Reflection; 
       3: namespace AssemblyLoad
       4: {    
       5:     class Program    
       6:     {        
       7:         static void Main(string[] args)        
       8:         {            
       9:             // 通过文件名加载程序集            
      10:             Assembly a = Assembly.LoadFrom("MyExe.exe");            
      11:             // 从程序集获得类型名称            
      12:             Type[] types2 = a.GetTypes();            
      13:             foreach (Type t in types2)            
      14:             {                
      15:                 Console.WriteLine(t.FullName);            
      16:             }             
      17:             Console.WriteLine();             
      18:             a = Assembly.LoadFrom("MyDll.dll");            
      19:             types2 = a.GetTypes();            
      20:             foreach (Type t in types2)            
      21:             {                
      22:                 Console.WriteLine(t.FullName);            
      23:             }            
      24:             Console.ReadKey();        
      25:         }    
      26:     }
      27: }

    在获取一个 Type 后,您可以采用许多方法发现与该类型的成员有关的信息。 例如,通过调用 Type.GetMembers 方法(该方法将获取对当前类型每个成员描述的一组 MemberInfo 对象),找到有关该类型的所有成员的信息。
    您也可以在 Type 类上使用方法,以检索有关按名称指定的一个或多个构造函数、方法、事件、字段或属性的信息。 例如,Type.GetConstructor 封装当前类的构造函数。
    如果具有 Type,则可以使用 Type.Module 属性来获取一个封装该类型所在模块的对象。 使用 Module.Assembly 属性可查找封装模块所在程序集的对象。 使用 Type.Assembly 属性可直接获取封装该类型的程序集。

    System.Type 和ConstructorInfo

    下面的示例演示如何列出一个类(此示例中为 String 类)的构造函数。

       1: using System;
       2: using System.Reflection; 
       3: namespace ListConstructors
       4: {    
       5:     class Program    
       6:     {        
       7:         static void Main(string[] args)        
       8:         {            
       9:             Type t = typeof(System.String);            
      10:             Console.WriteLine("正在列出类型 {0} 所有的 public 构造函数", t);            
      11:             ConstructorInfo[] ci = t.GetConstructors(BindingFlags.Public | BindingFlags.Instance);            
      12:             Console.WriteLine("//构造函数");            
      13:             PrintMembers(ci);             
      14:             Console.ReadKey();        
      15:         }        
      16:         public static void PrintMembers(MemberInfo[] ms)        
      17:         {            
      18:             foreach (MemberInfo m in ms)            
      19:             {                
      20:             Console.WriteLine("{0}{1}", "     ", m);            
      21:             }            
      22:             Console.WriteLine();        
      23:         }    
      24:     }
      25: }

    MemberInfo、MethodInfo、FieldInfo 和PropertyInfo

    使用 MemberInfo、MethodInfo、FieldInfo 或 PropertyInfo 对象获取有关该类型的方法、属性、事件和字段的信息。下面的示例使用 MemberInfo 列出 System.IO.File 类中的成员数量并使用 System.Type.IsPublic 属性确定该类的可见性。

       1: using System;
       2: using System.IO;
       3: using System.Reflection; 
       4: namespace Mymemberinfo
       5: {    
       6:     class Program    
       7:     {        
       8:         static void Main(string[] args)        
       9:         {            
      10:             Console.WriteLine("/nReflection.MemberInfo");            
      11:             Type MyType = Type.GetType("System.IO.File");            
      12:             MemberInfo[] Mymemberinfoarray = MyType.GetMembers();            
      13:             Console.WriteLine("/nThere are {0} members in {1}.", Mymemberinfoarray.Length, MyType.FullName);            
      14:             Console.WriteLine("{0}.", MyType.FullName);            
      15:             if (MyType.IsPublic)            
      16:             {                
      17:                 Console.WriteLine("{0} is public.", MyType.FullName);            
      18:             }            
      19:             Console.ReadKey();        
      20:         }    
      21:     }
      22: }

    下面的示例检查指定成员的类型,检查FieldInfo 类的GetValue 方法的信息。对 MemberInfo 类的一个成员执行反射,然后列出其类型。

       1: using System;
       2: using System.Reflection; 
       3: namespace MyMethodInfo
       4: {    
       5:     class Program    
       6:     {        
       7:         static void Main(string[] args)        
       8:         {            
       9:             Console.WriteLine("Reflection.MethodInfo");            
      10:             Type MyType = Type.GetType("System.Reflection.FieldInfo");            
      11:             // 指定您想要获得类型信息的成员 GetValue            
      12:             MethodInfo Mymethodinfo = MyType.GetMethod("GetValue");            
      13:             Console.WriteLine(MyType.FullName + "." + Mymethodinfo.Name);            
      14:             // 获取并显示 MemberType 属性            
      15:             MemberTypes Mymembertypes = Mymethodinfo.MemberType;            
      16:             switch (Mymembertypes)            
      17:             {                
      18:                 case MemberTypes.Constructor:                    
      19:                     Console.WriteLine("MemberType is of type All");                    
      20:                     break;                
      21:                 case MemberTypes.Custom:                    
      22:                     Console.WriteLine("MemberType is of type Custom");                    
      23:                     break;                
      24:                 case MemberTypes.Event:                    
      25:                     Console.WriteLine("MemberType is of type Event");                    
      26:                     break;                
      27:                 case MemberTypes.Field:                    
      28:                     Console.WriteLine("MemberType is of type Field");                    
      29:                     break;                
      30:                 case MemberTypes.Method:                    
      31:                     Console.WriteLine("MemberType is of type Method");                    
      32:                     break;                
      33:                 case MemberTypes.Property:                    
      34:                     Console.WriteLine("MemberType is of type Property");                    
      35:                     break;                
      36:                 case MemberTypes.TypeInfo:                    
      37:                     Console.WriteLine("MemberType is of type TypeInfo");                    
      38:                     break;            
      39:             }             
      40:             Console.ReadKey();        
      41:         }    
      42:     }
      43: }

    下面的示例使用反射 *Info 类以及 BindingFlags 来列出指定类的所有成员(构造函数、字段、属性、事件和方法),并将这些成员分为为静态和实例。

       1: using System;
       2: using System.Reflection; 
       3: namespace ListMembers
       4: {    
       5:     class Program    
       6:     {        
       7:         static void Main(string[] args)        
       8:         {            
       9:             // 指定类            
      10:             Type t = typeof(System.IO.BufferedStream);            
      11:             Console.WriteLine("正在列出 {0} 类型所有的成员(public 和 non public)...", t);             
      12:             // 列出 static 字段            
      13:             FieldInfo[] fi = t.GetFields(BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public);            
      14:             Console.WriteLine("// Static Fields");            
      15:             PrintMembers(fi);             
      16:             // 列出 Static 属性            
      17:             PropertyInfo[] pi = t.GetProperties(BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public);            
      18:             Console.WriteLine("// Static Properties");            
      19:             PrintMembers(pi);             
      20:             // 列出 Static 事件            
      21:             EventInfo[] ei = t.GetEvents(BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public);            
      22:             Console.WriteLine("// Static Events");            
      23:             PrintMembers(ei);             
      24:             // 列出 Static 方法            
      25:             MethodInfo[] mi = t.GetMethods(BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public);            
      26:             Console.WriteLine("// Static Methods");            
      27:             PrintMembers(mi);             
      28:             // 列出 Constructors            
      29:             ConstructorInfo[] ci = t.GetConstructors(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);            
      30:             Console.WriteLine("// Constructors");            
      31:             PrintMembers(ci);             
      32:             // 列出 Instance 字段            
      33:             fi = t.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);            
      34:             Console.WriteLine("// Instance Fields");            
      35:             PrintMembers(fi);             
      36:             // 列出 Instance 属性            
      37:             pi = t.GetProperties(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);            
      38:             Console.WriteLine("// Instance Properties");            
      39:             PrintMembers(pi);             
      40:             // 列出 Instance 事件            
      41:             ei = t.GetEvents(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);            
      42:             Console.WriteLine("// Instance Events");            
      43:             PrintMembers(ei);             
      44:             // 列出 Instance 方法            
      45:             mi = t.GetMethods(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);            
      46:             Console.WriteLine("// Instance Methods");            
      47:             PrintMembers(mi);             
      48:             Console.ReadKey();        
      49:         }         
      50:         public static void PrintMembers(MemberInfo[] ms)        
      51:         {            
      52:             foreach (MemberInfo m in ms)            
      53:             {                
      54:                 Console.WriteLine("{0}{1}", "     ", m);            
      55:             }            
      56:             Console.WriteLine();        
      57:         }    
      58:     }
      59: }
  • 相关阅读:
    Java Number & Math 类
    excel小技巧-转置(表的横向纵向倒过来)
    excel小技巧-提取填充
    excel小技巧-分列
    Java switch case语句
    if...else
    12.15个人总结
    12.9个人总结
    11.29个人总结
    11.25日个人总结
  • 原文地址:https://www.cnblogs.com/liuning8023/p/2131460.html
Copyright © 2011-2022 走看看