C#怎么实现AOP编程 C#面向切面编程方法

C#不原生支持AOP,但可通过四种方式实现:1. Aspect Injector(编译时织入,推荐新手);2. Castle DynamicProxy(运行时代理,适合IoC集成);3. Source Generators(.NET 6+编译时增强,性能最优);4. 装饰器模式(DI集成,简单可控)。

C#本身不原生支持AOP(面向切面编程),但可以通过多种成熟方式实现类似功能,核心思路是**在不修改业务代码的前提下,动态织入横切逻辑(如日志、权限、事务、异常处理等)**。常用且实用的方法有以下几种:

使用Aspect Injector(推荐新手)

这是一个轻量、开源、基于编译时织入的AOP库,语法简洁,无需配置复杂框架。

  • 安装NuGet包:Install-Package AspectInjector.Broker
  • 定义切面(如日志):
    [Aspect(Scope.Global)]
    public class LoggingAspect
    {
        [Advice(Kind.Before, Targets = Target.Method)]
        public void LogBefore([Argument(Source.Target)] object target,
                              [Argument(Source.Signature)] string signature)
        {
            Console.WriteLine($"[Before] {signature} on {target.GetType().Name}");
        }
    }
  • 在目标方法上打标签:[Inject] 即可自动织入

借助Castle DynamicProxy(运行时代理)

适用于需要对类或接口做拦截的场景,尤其适合IoC容器(如Autofac、DryIoc)集成。

  • 定义拦截器:
    public class LoggingInterceptor : IInterceptor
    {
        public void Intercept(IInvocation invocation)
        {
            Console.WriteLine($"Calling {invocation.Method.Name}");
            invocation.Proceed(); // 执行原方法
            Console.WriteLine($"Finished {invocation.Method.Name}");
        }
    }
  • 创建代理实例(需目标类实现接口,或为虚方法):
    var proxy = new ProxyGenerator()
        .CreateInterfaceProxyWithTarget(service, new LoggingInterceptor());

利用.NET 6+ Source Generators(编译时增强)

更现代、零运行时开销的方式,适合定制化强、性能敏感的场景。

  • 编写Source Generator,在编译时分析语法树,自动为标记方法插入日志/校验等代码
  • 需单独项目(Microsoft.CodeAnalysis.*引用),配合[GeneratedCode]或自定义特性触发
  • 学习成本略高,但生成代码完全透明、调试友好

结合依赖注入 + 装饰器模式(最简单可控)

不依赖第三方库,适合中小项目快速落地,符合SOLID原则。

  • 定义接口和原始服务:
    public interface IOrderService { void PlaceOrder(Order o); }
    public class OrderService : IOrderService { public void PlaceOrder(Order o) { /* 实现 */ } }
  • 写装饰器:
    public class LoggingOrderService : IOrderService
    {
        private readonly IOrderService _inner;
        public LoggingOrderService(IOrderService inner) => _inner = inner;
        public void PlaceOrder(Order o)
        {
            Console.WriteLine("Before placing order");
            _inner.PlaceOrder(o);
            Console.WriteLine("After placing order");
        }
    }
  • 注册时链式装饰:services.Decorate()(用Scrutor等扩展)

基本上就这些。选哪种取决于项目规模、团队熟悉度和性能要求:小项目用装饰器最快上手;中大型项目建议Aspect Injector或DynamicProxy;追求极致性能和可控性可探索Source Generators。