1.需求:为项目中通讯PLC模块实现AOP,实现统一的日志打印,参数校验,方法执行时间统计
2.问题:①现有项目没有IOC容器,没法使用部分AOP库的方法注册到IOC,(注:如果要实现IOC对现有代码改动大,并且AOP只是针对部分模块实现)②要在尽量小的代码改动下实现
针对以上问题选择使用Castle.DynamicProxy
3.实现:
①项目中所有通讯PLC(西门子、三菱、汇川等)的类都实现了统一接口PlcConnect
②通过代理模式统一对外暴露一个代理类PlcConnectProxy
③网上有许多关于Castle.DynamicProxy是怎么使用的,这里就不一一描述了,说一下我是怎么用的。
我是创建了一个Interceptor的类继承了StandardInterceptor,也可以实现IInterceptor,都没问题,StandardInterceptor也是实现IInterceptor这个接口。
public class Interceptor : StandardInterceptor { protected override void PreProceed(IInvocation invocation) { // 方法执行前 } protected override void PerformProceed(IInvocation invocation) { try { // 方法执行 base.PerformProceed(invocation); } catch (Exception ex) { HandleException(ex); } } protected override void PostProceed(IInvocation invocation) { // 方法执行后 } private void HandleException(Exception ex) { // 方法异常捕获 } }
再定义一个静态类ProxyExtension创建拦截器
public static class ProxyExtension { public static T GetHslAOPProxy<T>(this T t) where T : class { return new ProxyGenerator().CreateClassProxy<T>(new Interceptor()); } }
然后代理类在new的时候.GetHslAOPProxy()就可以
new PlcConnectProxy().GetHslAOPProxy()
4.带来问题
new PlcConnectProxy().GetHslAOPProxy()
PlcConnectProxy写成了单例但在创建时会初始化两个对象,导致被代理类有了两个,而我的需求是要保证被代理类是一个单例,破坏了单例模式。
原因:.GetHslAOPProxy()在调用时会生成一个子类继承父类方法来实现AOP。
解决:将所有通讯PLC(西门子、三菱、汇川等)的类写为单例,保证被代理类类只会生成一个(但又有个问题:虽然被代理类确实只有一个了,但PlcConnectProxy还会有两个,一个父类,一个子类)