摆脱MarshalByRefObject和DynamicProxy显示接口
最近在尝试把系统权限管理功能独立出来,自然的想到通过aop对现有的功能进行拦截,以添加用户权限校验步骤。通常可能是使用RealProxy 和 MarshalByRefObject,通过Attribute来标注需要拦截的方法,其实也没什么问题。但是如果是Webform,又希望对Webform中的方法进行拦截就有问题了,不支持类的多继承,MarshalByRefObject就不能用了,所以改用DynamicProxy,但问题又来了,总不能让每个Page都定义接口啊,这样会让现有的程序必须改动。DynamicProxy本身都挺好,就是这个显示接口麻烦,如果去掉这个显示接口就可以像RealProxy一样,方便的对方法进行代理了,(当然也可以用4.0中的dynamic完成对代理对象成员的调用)。
DynamicProxy中使用显示接口的作用有两个,一个是通过Emit直接实现接口,创建代理类型;二是在调用时可以使用接口来完成对代理成员的访问,在IDE中也可以有Intelligence提示。如果可以用被代理类型直接做为代理类型的父类,一切就都好办了。
办法分两步,一,提取代理类型成员签名,直接用Emit生成接口,用生成接口代替了显示定义接口;二,在代理类型创建时直接使用被代理类型做父类。被代理成员要用virtual标识。(最新尝试,不生成接口,直接用被代理对象类型为生成的代理的父类型就可以了,类似Castle.DynamicProxy)。
typeBuilder.SetParent(objType);//objType 被代理对象类型 //或者在DefineType时候直接指定parentType TypeBuilder typeBuilder = moduleBuilder.DefineType( dynamicTypeName, typeAttributes, objType, null);
在实用时候,可以保留接口部分。比如被代理类型为sealed,就换用object做父类型,同时实现接口(配合dynamic)或显示接口。
调用示例
public class Computer { public void TestDymaticProxy() { var computer = (Computer)DynamicProxy.SecurityProxy.NewInstance(new Computer()); computer.GetPrice2(); dynamic com = DynamicProxy.SecurityProxy.NewInstance(new Computer()); com.GetPrice2(); } public virtual void GetPrice() { Console.WriteLine("GetPrice"); } public virtual void GetPrice2() { Console.WriteLine("GetPrice2"); } }
https://files.cnblogs.com/heros/DynamicProxy.rar