C#如何实现AOP编程 Castle DynamicProxy使用教程

1次阅读

C#中实现AOP最成熟轻量的方案是Castle Dynamicproxy,它通过运行时代理生成实现方法拦截,支持接口和类代理,需实现IInterceptor接口并调用invocation.Proceed()执行原方法。

C#如何实现AOP编程 Castle DynamicProxy使用教程

在C#中实现AOP(面向切面编程),最成熟、轻量且广泛使用的方案是 Castle DynamicProxy。它不依赖编译器或IL修改,纯运行时通过代理生成实现方法拦截,适合日志、权限、事务、缓存等横切关注点。

一、安装与基础准备

使用 NuGet 安装核心包:

  • Castle.Core(含 DynamicProxy 实现)

注意:不需要安装 Castle.windsor(那是IOC容器,和AOP无关)。项目支持 .net Standard 2.0+ 或 .NET 5/6/7/8 均可。

二、定义拦截器(IInterceptor)

所有切面逻辑都封装在自定义拦截器中,必须实现 IInterceptor 接口:

public class LoggingInterceptor : IInterceptor {     public void Intercept(IInvocation invocation)     {         Console.WriteLine($"【开始】{invocation.Method.Name}");                  try         {             invocation.Proceed(); // 执行原方法                          Console.WriteLine($"【成功】{invocation.Method.Name} → {invocation.ReturnValue}");         }         catch (Exception ex)         {             Console.WriteLine($"【异常】{invocation.Method.Name} → {ex.Message}");             throw; // 不吞异常,保持行为一致         }     } }

关键点:

C#如何实现AOP编程 Castle DynamicProxy使用教程

Trae国内版

国内首款AI原生IDE,专为中国开发者打造

C#如何实现AOP编程 Castle DynamicProxy使用教程 2045

查看详情 C#如何实现AOP编程 Castle DynamicProxy使用教程

  • invocation.Proceed() 必须调用,否则原方法不会执行
  • 可读写 invocation.Argumentsinvocation.ReturnValue
  • 支持同步/异步拦截(需配合 IAsyncInterceptor,.NET Core 3.0+)

三、为类或接口创建代理

DynamicProxy 支持两种代理方式,适用场景不同:

  • 接口代理:目标类实现接口,用 CreateInterfaceProxyWithTarget(推荐,无侵入)
  • 类代理:目标类需有虚方法(virtual),用 CreateClassProxyWithTarget

示例(接口方式):

public interface IService {     string DoWork(); }  public class ServiceImpl : IService {     public virtual string DoWork() => "done"; }  // 创建代理 var proxyGenerator = new ProxyGenerator(); var interceptor = new LoggingInterceptor(); var target = new ServiceImpl(); var proxy = proxyGenerator.CreateInterfaceProxyWithTarget<IService>(target, interceptor);  Console.WriteLine(proxy.DoWork()); // 触发拦截器

四、进阶技巧与注意事项

实际项目中常遇到的问题和解法:

  • 多个拦截器:按添加顺序执行,传入数组:new[] { logInter, authInter }
  • 按方法名/特性过滤:在 Intercept 中检查 invocation.Method.GetCustomAttribute()
  • 避免代理“密封类”或“非虚方法”:类代理失败会静默回退到目标对象(无拦截),建议开启调试日志或单元测试验证
  • 生命周期管理:拦截器实例通常应是单例(如 LoggingInterceptor 无状态),避免在拦截器里持有上下文引用导致内存泄漏

基本上就这些。DynamicProxy 不复杂但容易忽略细节——重点是选对代理方式、确保方法可被拦截、拦截器别吞异常。用熟之后,搭配 DI 容器(如 microsoft.Extensions.DependencyInjection)自动注入代理,就能无缝集成到业务中。

text=ZqhQzanResources