C#C# Aop不难扫除文盲及OLX570M实体类属性拦截示例

先说下场景,C#中为什么要接纳Aop,而笔者又是在何地使用Aop?

本人只是想拦截实体类的Set的法门,然后在Set在此以前,调用一下别的措施,把值赋给另1个对象。

 

而自笔者做的皆以在实体类的基类里处理:

比如:

public class OrmBase

 

让抱有继续这一个基类的实体类都具备Orm操作成效,再添加一个非常小特殊的渴求处理,属性Set时,必要对另一对象赋值。

 

假使说,小编这么完毕:在OrmBase中可以提供格局,让抱有的子类的品质都如此操作:

 

public class Users:OrmBase

{

public int _ID;

public int ID 

{

get;

set

{

  base.SetXX(value);

 }

}

 

然则种种实体都如此写,就算是啥没难点,但是能简化的依旧简化。

 

在能追求简单的社会风气里,当然更欣赏简洁的写法如:

public
int ID {get;set;}

由此,直接在基类里从来拦截子类set方法,在里面平素调用SetXX就解决了,怎样落到实处呢?又花了一天的光阴查资料研商学习并达成。

 

C#,为此,要阻拦,就得折腾Aop:

古板的Aop使用RealProxy,使用非常简单,可是被忽悠的非凡复杂,上面:

 

1:在要阻拦的类头上加个属性标识,同时继续自ContextBoundObject:

 

[AopAttribute]

public class OrmBase:ContextBoundObject

 

OK,在基类里加2个,那样具有子类也算被增大了,加上一个标识,就足以被阻挡了,那这几个AopAttribute属性是什么?看上边

 

2:AopAttribute继承代理属性标识类,用来挂在要堵住的类的头上:

 

    class AopAttribute : ProxyAttribute

    {

        public override MarshalByRefObject CreateInstance(Type
serverType)

        {

            AopProxy realProxy = new AopProxy(serverType);

            return realProxy.GetTransparentProxy() as
MarshalByRefObject;

        }

    }

 

看,里面就两行,非凡不难,中间调用了继承RealProxy的AopProxy类,AopProxy是怎样,怎么出来的?看下面

 

3:AopProxy类,正是阻止的新闻处理,先上个简单版,免的大家看不懂:

 

 class AopProxy : RealProxy

    {

        public AopProxy(Type serverType)

            : base(serverType)

        {

        }

        public override IMessage Invoke(IMessage msg)

        {

            //新闻拦截之后,就会实施那里的措施。

        }

    }

OK,简单吗,就那样多个类,就能够兑现拦阻了,可是关键正是此处阻止之后的代码,稍为复杂点,一般照抄就行了,拦截的代码如下:

 if (msg is IConstructionCallMessage) //
如若是构造函数,按原来的艺术赶回即可。

            {

                IConstructionCallMessage constructCallMsg = msg as
IConstructionCallMessage;

                IConstructionReturnMessage constructionReturnMessage =
this.InitializeServerObject((IConstructionCallMessage)msg);

                RealProxy.SetStubData(this,
constructionReturnMessage.ReturnValue);

                return constructionReturnMessage;

            }

            else if (msg is IMethodCallMessage)
//假若是方法调用(属性也是办法调用的一种)

            {

                IMethodCallMessage callMsg = msg as IMethodCallMessage;

                object[] args = callMsg.Args;

                IMessage message;

                try

                {

                    if (callMsg.MethodName.StartsWith(“set_”) &&
args.Length == 1)

                    {

                       
//那里检查和测试到是set方法,然后应怎么调用对象的任何方法吗?

                    }

                    object o =
callMsg.MethodBase.Invoke(GetUnwrappedServer(), args);

                    message = new ReturnMessage(o, args, args.Length,
callMsg.LogicalCallContext, callMsg);

                }

                catch (Exception e)

                {

                    message = new ReturnMessage(e, callMsg);

                }

                return message;

            }

            return msg;

为了调用原始对象的任何方式,作者花了近一天的时日查资料,可惜网络上并没有对应的新闻,多数的人利用,都以引向1个别样方式(三个不供给调用原始对象的法门)

此时此刻互联网上Aop消息太少,C#的更少,关于什么收获原始对象,然后调用原始对象的,找不到一篇相关小说,作者特纠结。 

于是,笔者按古板办法,想尽办法的想取获得原来对象,再调用,经过九九八十一招,仍然败诉了。

(一发端是想:通过反射从品类再成立一个实体那种不可相信的品尝:
造成死循环,每一遍new拦截,在阻止里又new)

中间省一大堆……痛心的阅历和尝试…….

若果用思想,方法总有的,最终还是被作者发觉了:

1:获取要调用的法门:

在构造函数中,依据传进来的serverType,获取到SetXX的艺术MethodInfo:

method = serverType.GetMethod(“SetXX”, BindingFlags.NonPublic |
BindingFlags.Instance);

2:在阻止方法中调用:

 if (callMsg.MethodName.StartsWith(“set_”) && args.Length == 1)

{

   method.Invoke(GetUnwrappedServer(), new object[] {
callMsg.MethodName.Substring(4), args[0] });//对质量进行调用

  }

进度很复杂,尝试过N百种格局,结果非常粗略,分享很要紧!

为此,化解了O汉兰达M对子类的品质拦截,并实现了在性质赋值时调用实例别的方式。

相关文章