反射是C#在运行时动态查看、检查和操作类型及成员的能力,依赖System.Type等类实现,关键在于“运行时”和“动态”,支持加载程序集、创建实例、调用方法;常用Type获取方式包括typeof、GetType()、Type.GetType()和Assembly.GetType();可通过GetMethods()等方法发现成员并用BindingFlags筛选;支持Activator.CreateInstance()创建对象、Invoke()调用方法、GetValue()/SetValue()读写属性或字段;但性能较低,需注意异常处理、权限限制及混淆影响。

反射是C#在运行时查看、检查甚至操作类型、方法、属性、字段等元数据的能力。它不依赖编译期已知的类型,而是通过 System.Type 和相关类,在程序执行中动态发现和调用成员。关键在于“运行时”和“动态”——你不需要提前写死类名或方法名,也能加载程序集、创建实例、调用方法。
获取Type对象的几种常用方式
要使用反射,第一步是拿到 Type 实例:
-
typeof(MyClass)—— 编译期已知类型,最轻量、推荐用于本程序内类型 -
obj.GetType()—— 对已有实例获取其实际运行时类型(支持多态) -
Type.GetType("Namespace.ClassName")—— 通过完整字符串名称获取,需注意命名空间+程序集限定(如未指定,默认只查当前程序集) -
Assembly.GetExecutingAssembly().GetType("...")—— 显式从指定程序集中查找,适合插件或外部DLL场景
查看类型结构:成员发现与筛选
拿到 Type 后,可用一系列 GetXXX() 方法列出成员:
-
type.GetMethods()返回所有公共方法;加BindingFlags可控制可见性(如BindingFlags.NonPublic | BindingFlags.Instance查私有实例方法) -
type.GetProperties()、type.GetFields()、type.GetConstructors()同理 - 常用组合:
BindingFlags.Public | BindingFlags.Instance查公有实例成员;BindingFlags.Static | BindingFlags.FlattenHierarchy查静态继承成员 - 建议配合 LINQ 筛选,例如
type.GetMethods().Where(m => m.Name.StartsWith("Get"))
动态创建对象并调用成员
反射不仅看,还能做:
- 创建实例:
Activator.CreateInstance(type)(调用无参构造);或传入参数数组调用带参构造 - 调用方法:
methodInfo.Invoke(obj, args),第一个参数是目标实例(静态方法传null) - 读写属性:
propertyInfo.GetValue(obj)/propertyInfo.SetValue(obj, value) - 访问字段:
fieldInfo.GetValue(obj)/fieldInfo.SetValue(obj, value)(对私有字段也有效)
性能与安全注意事项
反射灵活但有代价:
- 比直接调用慢得多——JIT无法优化,每次都要解析元数据、校验权限、装箱拆箱。高频场景建议缓存
MethodInfo或用Delegate.CreateDelegate转为委托 - 绕过编译检查,容易在运行时报
TargetInvocationException或ArgumentException,务必做好 try-catch - .NET Core/.NET 5+ 默认禁用某些反射操作(如访问非公开成员),需确保运行时有对应权限(如
ReflectionPermission已废弃,但部分策略仍影响行为) - 混淆工具(如 ILLink、Dotfuscator)可能移除未显式引用的成员,导致反射失败,必要时用
[DynamicDependency]或PreserveAttribute标记
基本上就这些。反射不是日常首选,但在序列化、ORM、DI容器、测试模拟、插件系统等场景中不可替代——理解它怎么“看”和“动”,才能用得稳、改得准、查得清。










