在 .NET 4.5 中反射机制的变更

在 .NET 4.5 中反射机制的变更

反射机制(Reflection)通常会涉及到3中场景:

  • 运行时反射
    • 场景:可以检索已加载程序集、类型、对象、实例和方法调用的元数据(Metadata)。
    • .NET 支持情况:支持
  • 仅供静态分析的反射
    • 场景:可以检索程序集中定义的类型和对象的元数据,而不会产生其他副作用。
    • .NET 支持情况:有限的支持
  • 反射扩展性
    • 场景:在上述两种场景下扩展元数据。
    • .NET 支持情况:支持,但非常复杂

System.Type 是反射模型中的最主要的抽象和入口点。它被用于描述两种相关但却不同的概念:基本信息(reference)和定义(definition),并使两者之间可以互操作。这使得其并不满足关注点分离(Separation of Concerns)设计原则。

从概念上讲,reference 是某种事物的浅显表述,而 definition 提供更丰富的表述。例如 System.Reflection.Assembly 类代表了程序集的 definitions,而System.Reflection.AssemblyName 类则代表了程序集的 references 。前者暴露了丰富的功能,而后者仅是提供能够获取定义的数据。

.NET 4.5 中包含了一些对传统反射机制的变更。其中最重要的变化是,System.Type 类被分割至两个独立的类中:Type 和 TypeInfo。TypeInfo 实例包含 Type 的定义,而 Type 则只关心数据。

  • Type:提供对象结构的浅视图,主要用于持有数据。(reference
  • TypeInfo:提供对象结构的完整视图,包含对象与父类及子类的关系。(definition

BCL中的变化

  • Assembly.DefinedTypes 属性返回 TypeInfo。
    • 该 API 负责获取程序集内定义的所有类型。
  • Type.BaseType 属性返回 Type。
    • 该 API 尽返回父类型的表述,而不是 definition
    • 父类型有可能被定义在另一个程序集内,这就需要一次程序集加载。
  • Object.GetType 方法返回 Type。
    • 该 API 返回 Type,因为你仅需要类型的表述。
    • 该类型有可能被定义在另一个程序集内,这就需要一次程序集加载。
  • C# 关键字 typeof 返回 Type。
    • 与 Object.GetType 行为相同。

代码示例

如果你已经在基于 .NET 4.5 的桌面或者 Web 端应用程序中使用了反射机制,原有的 API 仍然可以使用,并且会与新的 API 共存,但新的 API 提供了更轻量级的实现。TypeInfo 使用 IEnumerable 泛型集合来构建元数据,而不是传统的数组。这使得 TypeInfo 可以支持使用 Lazy 方式遍历对象元数据,并且可以引入 LINQ 来使查询更加便捷。

我们先来看下使用传统 API 来检索 Type 元数据的方式。Type API 可以提供检索类型名称、名空间、类型全名称、所在模块等信息。API 结构与 .NET 4.0 中一致。

1       Type studentType = typeof(Student);
2       string studentAssembly = studentType.Assembly.FullName;
3       string studentFullName = studentType.FullName;
4       string studentNameSpace = studentType.Namespace;
5       string studentModule = studentType.Module.FullyQualifiedName;

在 .NET 4.5 中反射机制的变更

正如你所看到的,Type 提供了对象结构的基本视图。如果你需要对类的结构挖掘的更新一些,现在可以使用新的 TypeInfo API。可以通过 GetTypeInfo() 方法从 Type 获取其 TypeInfo 定义。

我们先定义一个 Student 类:

1   public class Student
2   {
3     public string Name { get; set; }
4     public int Class { get; set; }
5     public int GetTestReport() { return 100; }
6     public event EventHandler Running;
7   }

现在,假如说你需要查看 Student 类中定义的属性、方法、事件等,可以通过 TypeInfo 提供的 API 获得。

1       TypeInfo studentInfo = studentType.GetTypeInfo();
2       IEnumerable<PropertyInfo> declaredProperties = studentInfo.DeclaredProperties;
3       IEnumerable<MethodInfo> declaredMethods = studentInfo.DeclaredMethods;
4       IEnumerable<EventInfo> declaredEvents = studentInfo.DeclaredEvents;

在 .NET 4.5 中反射机制的变更

另一个常见的反射方式是查找程序集内定义的所有类型。在 API 变化之后,Assembly.DefinedTypes 将返回一个 TypeInfo 的集合,而不是 Type 数组。

1       Assembly thisAssembly = typeof(Student).GetTypeInfo().Assembly;
2       IEnumerable<TypeInfo> allDefinedTypes = thisAssembly.DefinedTypes;

TypeInfo 除了提供上面描述的属性,还提供了若干用于获取元数据信息的方法。

参考资料