news 2026/4/16 19:24:25

C# 反射

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C# 反射

C# 中的反射(Reflection)是 .NET 提供的一种强大机制,允许程序在运行时检查、分析和操作类型(类、结构、接口、方法、属性等)的元数据,并动态创建对象、调用方法、访问字段或属性。它主要通过System.Reflection命名空间中的类实现。


一、反射的核心用途

  1. 动态加载程序集(Assembly)
  2. 获取类型信息(Type)
  3. 创建对象实例(Activator / ConstructorInfo)
  4. 调用方法(MethodInfo.Invoke)
  5. 读写属性或字段(PropertyInfo / FieldInfo)
  6. 检查特性(Attribute)
  7. 实现插件系统、ORM、序列化框架等

二、基础使用步骤

1. 获取 Type 对象

// 方法1:通过 typeofTypetype1=typeof(string);// 方法2:通过对象的 GetType()stringstr="hello";Typetype2=str.GetType();// 方法3:通过类型全名从 Assembly 加载Typetype3=Type.GetType("System.String");// 方法4:从程序集加载Assemblyassembly=Assembly.LoadFrom("MyLibrary.dll");Typetype4=assembly.GetType("MyNamespace.MyClass");

注意:Type.GetType("...")只能加载当前应用域中已加载或 mscorlib/System 中的类型,跨程序集需指定完整名称(含 Assembly 信息)。


2. 创建对象实例

使用Activator.CreateInstance
Typetype=typeof(Person);objectobj=Activator.CreateInstance(type);// 调用无参构造函数// 带参数构造objectobj2=Activator.CreateInstance(type,"Alice",30);
使用ConstructorInfo
ConstructorInfoctor=type.GetConstructor(newType[]{typeof(string),typeof(int)});objectinstance=ctor.Invoke(newobject[]{"Bob",25});

3. 调用方法

Typetype=typeof(Calculator);objectcalc=Activator.CreateInstance(type);// 获取方法MethodInfomethod=type.GetMethod("Add");// 调用实例方法intresult=(int)method.Invoke(calc,newobject[]{10,20});// 调用静态方法MethodInfostaticMethod=type.GetMethod("Multiply");intstaticResult=(int)staticMethod.Invoke(null,newobject[]{5,6});

如果方法是泛型,需先调用MakeGenericMethod()


4. 访问属性(Property)

Typetype=typeof(Person);objectperson=Activator.CreateInstance(type);// 设置属性PropertyInfonameProp=type.GetProperty("Name");nameProp.SetValue(person,"Charlie");// 获取属性值stringname=(string)nameProp.GetValue(person);

5. 访问字段(Field)

FieldInfofield=type.GetField("_age",BindingFlags.NonPublic|BindingFlags.Instance);field.SetValue(person,28);intage=(int)field.GetValue(person);

注意:私有成员需要BindingFlags.NonPublic


6. 获取所有成员

Typetype=typeof(MyClass);// 所有公共方法MethodInfo[]methods=type.GetMethods();// 所有公共属性PropertyInfo[]props=type.GetProperties();// 所有字段(包括私有)FieldInfo[]fields=type.GetFields(BindingFlags.Public|BindingFlags.NonPublic|BindingFlags.Instance);// 所有构造函数ConstructorInfo[]ctors=type.GetConstructors();

7. 检查和读取特性(Attribute)

[DisplayName("用户实体")]publicclassUser{}// 检查是否有某个 AttributeboolhasAttr=type.IsDefined(typeof(DisplayNameAttribute),false);// 获取 Attribute 实例varattr=(DisplayNameAttribute)type.GetCustomAttribute(typeof(DisplayNameAttribute));Console.WriteLine(attr.DisplayName);// 输出:"用户实体"

三、常用 BindingFlags 枚举值

标志说明
Public公共成员(默认)
NonPublic私有/受保护成员
Instance实例成员
Static静态成员
DeclaredOnly仅当前类型声明的成员(不包括继承)

示例:

type.GetMethod("PrivateMethod",BindingFlags.NonPublic|BindingFlags.Instance);

四、性能注意事项

  • 反射比直接调用慢很多(可能慢 10~100 倍),因为涉及类型查找、安全检查、装箱/拆箱等。
  • 避免在高频循环中使用反射
  • 可缓存 MethodInfo / PropertyInfo 等以提升性能。
  • 考虑使用委托(如Delegate.CreateDelegate)或表达式树(Expression)优化
  • .NET Core / .NET 5+ 引入了System.Reflection.Emitsource generator等更高效替代方案。

五、实际应用场景举例

1. 通用对象拷贝器

publicstaticvoidCopyProperties(objectsource,objecttarget){varsourceType=source.GetType();vartargetType=target.GetType();foreach(varpropinsourceType.GetProperties()){vartargetProp=targetType.GetProperty(prop.Name);if(targetProp!=null&&targetProp.CanWrite){targetProp.SetValue(target,prop.GetValue(source));}}}

2. 插件系统

Assemblyplugin=Assembly.LoadFrom("Plugin.dll");TypepluginType=plugin.GetTypes().FirstOrDefault(t=>typeof(IPlugin).IsAssignableFrom(t));IPlugininstance=(IPlugin)Activator.CreateInstance(pluginType);instance.Execute();

3. ORM 映射(如将 DataTable 转为对象)

publicstaticTToObject<T>(DataRowrow)whereT:new(){Tobj=newT();Typetype=typeof(T);foreach(DataColumncolinrow.Table.Columns){PropertyInfoprop=type.GetProperty(col.ColumnName);if(prop!=null&&row[col]!=DBNull.Value)prop.SetValue(obj,row[col]);}returnobj;}

六、总结

优点缺点
高度灵活,支持运行时动态行为性能开销大
实现通用框架(如 DI、AOP、序列化)代码可读性降低
支持插件、脚本扩展容易引发运行时异常(如拼写错误)
可用于调试、测试工具不支持 AOT 编译(如 Native AOT)的部分场景

建议:仅在必要时使用反射,并做好异常处理(如NullReferenceExceptionTargetExceptionMissingMethodException等)。


版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 7:21:47

深度强化学习:表格Q-network

一、价值迭代的问题在FrozenLake环境中&#xff0c;交叉熵方法改为价值迭代后&#xff0c;模型收敛速度加快。价值迭代时对所有状态进行循环&#xff0c;并对每个状态用Bellman方程更新价值。该过程中&#xff0c;对于同一方法中Q值&#xff08;动作价值&#xff09;变化似乎相…

作者头像 李华
网站建设 2026/4/16 7:22:14

JavaScript 装饰器完全指南(原理/分类/场景/实战/兼容)

JavaScript 装饰器&#xff08;Decorator&#xff09;是 ES7 提案中的特性&#xff0c;核心是通过“包装目标对象”&#xff0c;在不修改原对象源码的前提下&#xff0c;动态扩展其功能&#xff0c;本质是“高阶函数的语法糖”&#xff0c;让代码复用、功能增强更简洁优雅&…

作者头像 李华
网站建设 2026/4/16 7:10:26

解锁Qwen3-8B全部潜力:32K上下文窗口的实际应用场景解析

解锁Qwen3-8B全部潜力&#xff1a;32K上下文窗口的实际应用场景解析 在智能客服反复忘记用户上一轮诉求、代码助手只能看到函数片段而误判逻辑、企业知识库问答总是“断章取义”的今天&#xff0c;我们不得不面对一个现实&#xff1a;大多数语言模型的“记性”太差。它们或许能…

作者头像 李华
网站建设 2026/4/16 7:26:36

在 Windows 11 上恢复已删除文件的 10 大方法 [2025]

如果您不小心删除了 Windows 11 中的文件怎么办&#xff1f;不用担心&#xff0c;本文提供了 10 种有效的方法来逐步恢复 Windows 11 上已删除的文件。向下滚动以了解更多信息。 Windows 11 是 Microsoft 的最新作系统&#xff0c;在利用下一代处理器的计算和内存能力的同时&am…

作者头像 李华
网站建设 2026/4/16 9:03:56

【ESP32-S3】运行报错集合

【ESP32-S3】运行报错集合报错assert failed: block_locate_free tlsf_control_functions.h:618 (block_size(block) > *size)参考报错 assert failed: block_locate_free tlsf_control_functions.h:618 (block_size(block) > *size) 成功启动HTTP服务器后&#xff0c;…

作者头像 李华
网站建设 2026/4/16 10:39:26

Transformer模型详解系列:Seed-Coder-8B-Base中的注意力机制应用

Transformer模型详解系列&#xff1a;Seed-Coder-8B-Base中的注意力机制应用 在现代软件开发中&#xff0c;一个再普通不过的场景是&#xff1a;程序员刚写完函数签名&#xff0c;还没来得及敲下一行逻辑代码&#xff0c;IDE就已经“预知”了接下来要实现的功能——自动补全变…

作者头像 李华