前言
前段时间有朋友问道一个这样的问题,.NET Core中如何通过Attribute的元数据信息来调用标记的对应方法。我第一时间想到的就是通过C#反射获取带有Custom Attribute标记的类,然后通过依赖注入(DI)的方式获取对应服务的方法并通过反射动态执行类的方法,从而实现更灵活的编程方式。
什么是反射?
在 C# 中,反射是指在运行时动态地获取类型的信息并操作对象的能力。使用反射,我们可以在代码中访问程序集、模块、成员等,并且可以操作这些成员的属性、方法、字段和事件等。
反射的作用
- 动态加载程序集。
- 获取类型信息。
- 创建对象和调用方法。
- 访问和操作成员。
- 扩展框架和库。
注意:由于反射是一种非常灵活和强大的机制,但也带来了一定的性能开销。因此,在使用反射时应慎重考虑其适用性,并权衡性能和灵活性的取舍。
自定义一个Attribute类型
/// <summary> /// 自定义一个Attribute类型 /// </summary> [AttributeUsage(AttributeTargets.Class, AllowMultiple = false)] public class CustomAttribute : Attribute { public string TargetMethod { get; set; } public CustomAttribute(string targetMethod) { TargetMethod = targetMethod; } }
定义如下两个需要被执行的服务,并使用CustomAttribute标记
/// <summary> /// 前进服务 /// </summary> [Custom("AdvanceWay")] public class AdvanceService { public void AdvanceWay() { Console.WriteLine("On the move!"); } } /// <summary> /// 后退服务 /// </summary> [Custom("RetreatWay")] public class RetreatService { public void RetreatWay() { Console.WriteLine("Be retreating!"); } }
注册需要注入的服务
var services = new ServiceCollection(); //注册需要注入的服务 services.AddTransient<AdvanceService>(); services.AddTransient<RetreatService>();
反射获取所有带有CustomAttribute特性的类并调用对应方法
static void Main(string[] args) { var services = new ServiceCollection(); //注册需要注入的服务 services.AddTransient<AdvanceService>(); services.AddTransient<RetreatService>(); var provider = services.BuildServiceProvider(); #region 反射获取所有带有CustomAttribute特性的类并调用对应方法 //反射获取所有带有CustomAttribute特性的类 var classes = Assembly.GetExecutingAssembly().GetTypes() .Where(type => type.GetCustomAttributes<CustomAttribute>().Any()); foreach (var clazz in classes) { //获取标记CustomAttribute的实例 var attr = clazz.GetCustomAttributes<CustomAttribute>().First(); //根据CustomAttribute元数据信息调用对应的方法 var methodInfo = clazz.GetMethod(attr.TargetMethod); if (methodInfo != null) { //instance 对象是通过依赖注入容器获取的。这是一种常用的实现方式,可以使用依赖注入解耦程序中各个组件之间的依赖关系,方便测试和维护。 var instance = provider.GetService(clazz); methodInfo.Invoke(instance, null); } } #endregion #region 反射获取所有带有CustomAttribute特性的类并调用指定方法 var executionMethod = "RetreatWay"; foreach (var clazz in classes) { //获取标记CustomAttribute的实例 var attr = clazz.GetCustomAttributes<CustomAttribute>().First(); if (attr.TargetMethod == executionMethod) { //根据CustomAttribute元数据信息调用对应的方法 var methodInfo = clazz.GetMethod(attr.TargetMethod); if (methodInfo != null) { //instance 对象是通过依赖注入容器获取的。这是一种常用的实现方式,可以使用依赖注入解耦程序中各个组件之间的依赖关系,方便测试和维护。 var instance = provider.GetService(clazz); methodInfo.Invoke(instance, null); } } } #endregion Console.ReadLine(); }
输出如下: