C#不原生支持AOP,但可通过四种方式实现:1. Aspect Injector(编译时织入,推荐新手);2. Castle DynamicProxy(运行时代理,适合IoC集成);3. Source Generators(.NET 6+编译时增强,性能最优);4. 装饰器模式(DI集成,简单可控)。
C#本身不原生支持AOP(面向切面编程),但可以通过多种成熟方式实现类似功能,核心思路是**在不修改业务代码的前提下,动态织入横切逻辑(如日志、权限、事务、异常处理等)**。常用且实用的方法有以下几种:
这是一个轻量、开源、基于编译时织入的AOP库,语法简洁,无需配置复杂框架。
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] 即可自动织入适用于需要对类或接口做拦截的场景,尤其适合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()); 更现代、零运行时开销的方式,适合定制化强、性能敏感的场景。
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。