反射(Reflection)是C#中的一种机制,允许程序在运行时获取类型信息并动态调用其成员。通过反射,程序可以访问程序集、模块、类型及其成员(如方法、属性、字段等),并能在运行时创建对象、调用方法或访问属性。
反射的核心类是 System.Type
,它表示类型声明(类、接口、数组、值类型等)。通过 Type
对象,可以获取类型的详细信息并执行相关操作。
反射的主要功能
-
获取类型信息:通过
typeof
或GetType()
获取Type
对象。 -
动态创建对象:使用
Activator.CreateInstance
方法。 -
调用方法:通过
MethodInfo
调用方法。 -
访问字段和属性:通过
FieldInfo
和PropertyInfo
访问字段和属性。 -
加载程序集:使用
Assembly.Load
或Assembly.LoadFrom
加载程序集。
反射的常见应用
-
插件系统:
-
通过反射动态加载程序集并调用其中的类型和方法,常用于插件架构。
csharp
Assembly pluginAssembly = Assembly.LoadFrom("MyPlugin.dll"); Type pluginType = pluginAssembly.GetType("MyPlugin.MyClass"); object pluginInstance = Activator.CreateInstance(pluginType); MethodInfo method = pluginType.GetMethod("DoWork"); method.Invoke(pluginInstance, null);
-
-
依赖注入:
-
反射用于在依赖注入容器中动态创建和注入对象。
csharp
Type serviceType = typeof(MyService); object serviceInstance = Activator.CreateInstance(serviceType);
-
-
序列化和反序列化:
-
反射用于在序列化和反序列化过程中访问对象的属性和字段。
csharp
foreach (PropertyInfo prop in obj.GetType().GetProperties()) {Console.WriteLine($"{prop.Name}: {prop.GetValue(obj)}"); }
-
-
ORM框架:
-
反射用于对象关系映射(ORM)框架中,动态读取和设置数据库记录的属性。
csharp
foreach (var prop in entity.GetType().GetProperties()) {prop.SetValue(entity, reader[prop.Name]); }
-
-
单元测试:
-
反射用于单元测试框架中,动态调用私有方法和访问私有字段。
csharp
MethodInfo privateMethod = typeof(MyClass).GetMethod("PrivateMethod", BindingFlags.NonPublic | BindingFlags.Instance); privateMethod.Invoke(myClassInstance, null);
-
-
动态代理:
-
反射用于创建动态代理对象,拦截方法调用并执行额外逻辑。
csharp
public class DynamicProxy : DispatchProxy {protected override object Invoke(MethodInfo targetMethod, object[] args){Console.WriteLine($"Before {targetMethod.Name}");var result = targetMethod.Invoke(target, args);Console.WriteLine($"After {targetMethod.Name}");return result;} }
-
反射的优缺点
优点
-
灵活性:允许在运行时动态操作类型和对象。
-
扩展性:支持插件和模块化设计。
缺点
-
性能开销:反射操作通常比直接代码调用慢。
-
安全性:可能绕过访问控制,带来安全隐患。
-
复杂性:代码可读性和维护性较差。
总结
反射是C#中强大的机制,适用于需要动态类型操作的场景,如插件系统、依赖注入等。尽管灵活,但需注意其性能和安全问题。