摆脱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

posted @ 2012-05-10 01:03  heros  阅读(615)  评论(0编辑  收藏  举报