转自: http://www.cnblogs.com/cyq1162/archive/2012/05/30/2526573.html

先说下场景,C#中为什么要使用Aop,而我又是在哪里使用Aop?

 
本人只是想拦截实体类的Set的方法,然后在Set之前,调用一下其它方法,把值赋给另一个对象。
 
而我做的都是在实体类的基类里处理:
 
比如:
 
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就搞定了,如何实现呢?又花了一天的时间查资料研究学习并实现。

 
为此,要拦截,就得折腾Aop:
 
传统的Aop使用RealProxy,使用非常简单,但是被忽悠的非常复杂,下面:
 
1:在要拦截的类头上加个属性标识,同时继承自ContextBoundObject:
 

[AopAttribute]
public class OrmBase:ContextBoundObject
 
 

OK,在基类里加一个,这样所有子类也算被附加了,加上一个标识,就可以被拦截了,那这个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;
 

为了调用原始对象的其它方法,我花了近一天的时间查资料,可惜网络上并没有相应的信息,多数的人应用,都是引向一个其它方法(一个不需要调用原始对象的方法)

目前网络上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百种方式,结果很简单,分享很重要!

为此,解决了ORM对子类的属性拦截,并实现了在属性赋值时调用实例其它方法。

(转)C# Aop简单扫盲及ORM实体类属性拦截示例的更多相关文章

  1. Mybatis系列(二):优化MyBatis配置文件中的配置和解决字段名与实体类属性名不相同的冲突

    原文链接:http://www.cnblogs.com/xdp-gacl/p/4264301.html     http://www.cnblogs.com/xdp-gacl/p/4264425.ht ...

  2. Android利用反射机制为实体类属性赋值

    在做android项目时,有时会遇到从网络上获取json类型数据,赋值给实体类,实体类属性少可以一个一个的赋值,如果实体类有很多属性,赋值可能就要耗很长的功夫了,幸好Java给我们提供了反射机制.下面 ...

  3. KO ------- 表中字段名和实体类属性名不一致

    -----------------------siwuxie095 KO ------- 表中字段名和实体类属性名不一致 如果数据库表中的字段名和实体类的属性名不一致,那么在查询时, 相应字段的结果就 ...

  4. Mybatis——实体类属性名和数据库字段名不同时的解决方案

    数据库的字段: 对应的实体类: 方案一: 在XML映射文件中使用的resultMap,优点:可以被重复使用. <resultMap id="BaseResultMap" ty ...

  5. Mybatis中实体类属性和数据列之间映射的四种办法

    http://blog.csdn.net/lmy86263/article/details/53150091 Mybatis不像hibernate中那么自动化,通过@Column注解或者直接使用实体类 ...

  6. Mybatis中实体类属性与数据库列表间映射方法介绍

               这篇文章主要介绍了Mybatis中实体类属性与数据列表间映射方法介绍,一共四种方法方法,供大家参考.         Mybatis不像Hibernate中那么自动化,通过@Co ...

  7. MyBatis学习总结(四)——解决字段名与实体类属性名不相同的冲突(转载)

    本文转载自:http://www.cnblogs.com/jpf-java/p/6013307.html 在平时的开发中,我们表中的字段名和表对应实体类的属性名称不一定都是完全相同的,下面来演示一下这 ...

  8. mybatis框架下解决数据库中表的列的字段名和实体类属性不相同的问题

    导包.... 实体类中的属性,getter,setter,tostring,构造等方法就不写了 private int id; private String orderNo; private floa ...

  9. MyBatis入门学习教程-解决字段名与实体类属性名不相同的冲突

    在平时的开发中,我们表中的字段名和表对应实体类的属性名称不一定都是完全相同的,下面来演示一下这种情况下的如何解决字段名与实体类属性名不相同的冲突. 一.准备演示需要使用的表和数据 CREATE TAB ...

随机推荐

  1. gcc 动态编译 动态库路径

    gcc 动态编译(共享库) 动态编译的可执行文件需要附带一个的动态链接库,在执行时,需要调用其对应动态链接库中的命令优点:体积小,编译快缺点:依赖性高 代码如下: [root@74-82-173-21 ...

  2. springboot 定制错误页面

    项目中经常遇到的异常情况 400-Bad Request 401-Unauthorized If the request already included Authorization credenti ...

  3. Linux学习笔记(第十一章)

    文件系统及程序资源的配置ulimit: 环境变量: bash变量 alias设定变量别名 设定别名 取消别名 指令执行顺序 组合键 通配符 数据流重导向 多指令 以下命令都需用管道符链接: 截取命令: ...

  4. 20155203 杜可欣《网络对抗技术》Exp1 PC平台逆向破解

    1.1 实践目标 本次实践的对象是一个名为pwn1的linux可执行文件. 该程序正常执行流程是:main调用foo函数,foo函数会简单回显任何用户输入的字符串. 该程序同时包含另一个代码片段,ge ...

  5. VS编程,WPF中两个滚动条 ScrollViewer 同步滚动的一种方法

    原文:VS编程,WPF中两个滚动条 ScrollViewer 同步滚动的一种方法 版权声明:我不生产代码,我只是代码的搬运工. https://blog.csdn.net/qq_43307934/ar ...

  6. VS编程,编辑WPF过程中,点击设计器中界面某一控件,在XAML中高亮突出显示相应的控件代码的设置方法。

    原文:VS编程,编辑WPF过程中,点击设计器中界面某一控件,在XAML中高亮突出显示相应的控件代码的设置方法. 版权声明:我不生产代码,我只是代码的搬运工. https://blog.csdn.net ...

  7. 汇编 do while循环

    do while生成的汇编代码  do while汇编还原成C++代码 一. do while成生的汇编代码 // int i=0; // do // { // i++; // } while ( ...

  8. linux下的yum命令详细介绍

    yum(全称为 Yellow dog Updater, Modified)是一个在Fedora和RedHat以及SUSE中的Shell前端软件包管理器.基於RPM包管理,能够从指定的服务器自动下载RP ...

  9. MFC学习笔记(一): 不用MFC向导如何新建一个MFC程序

    使用Visual Studio新建一个空项目,项目命名为HelloMFC,完成后,打开项目属性页面,将配置属性选项卡中的常规项打开,将其中的MFC的使用属性栏改为:在静态库中使用MFC或者在共享DLL ...

  10. Kubernetes采用CoreDNS

    参考文档: kubernetes插件:https://github.com/kubernetes/kubernetes/tree/master/cluster/addons/dns/coredns 自 ...