C# Aop简单扫盲及ORM实体类属性拦截示例

先说下场景,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对子类的属性拦截,并实现了在属性赋值时调用实例其它方法。

时间: 2024-10-29 10:17:55

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

MyBatis学习教程(四)-如何快速解决字段名与实体类属性名不相同的冲突问题_java

在项目开发中,我们经常会遇到表中的字段名和表对应实体类的属性名称不一定都是完全相同的情况,下面小编给大家演示一下这种情况下的如何解决字段名与实体类属性名不相同的冲突问题,感兴趣的朋友一起学习吧. 一.准备演示需要使用的表和数据 CREATE TABLE orders( order_id INT PRIMARY KEY AUTO_INCREMENT, order_no VARCHAR(20), order_price FLOAT ); INSERT INTO orders(order_no, or

发布一个实体类属性生成小工具,给开发加点料

做了很久的代码生成工具,基本上都是基于表生成实体类属性的,把数据库表的信息拿出来,然后之乎者也后生成一个标准的实体类,包含字段.属性.描述等东西. 是基于整个数据库生成整个框架代码的工具,如我的代码生成工具Database2Sharp(下载地址http://www.iqidi.com/Download/Database2SharpSetup.rar ),基本上能满足前期的框架代码生成. 不过后来在做一些非数据库的项目的实体类,还有一些不是基于表一一对应关系的实体类,写这些字段属性的代码就显得比较

.net中如何给数据表实体类赋值的方法

数据 今天遇到一个问题,就是数据实体类中如何给属性赋值.因为从数据库中取出来的是一个DataSet如果一个一个给数据实体类中的属性赋值显的非常的笨拙,而且我们的数据表多达24个.在网上搜索了下写出了个简单的方法,代码如下: 首先我们做个实体类  1     public class Class1 2     { 3         private int inttemp; 4  5         public int IntTemp 6         { 7             get

EF code first 实体类修改、添加、删除操作问题

问题描述 EF code first 实体类修改.添加.删除操作问题 EF code First 怎么来更新已存在的数据库结构? 比如:在实体类中我新添加了一个字段.修改字段名称或者类型.删除一个字段 等操作后我怎么更新到已存在的数据库对于表中去. 担心:我在持续开发中,改变了数据库的结构,如果这时候用户已经用以前的版本创建好数据库并且有了许多数据,假使我的修改只增加了一个表的一个可空列.而为此我只能重新创建数据库.请问有没有别的办法,能够把新的数据库结构在不损伤数据的情况下更新到部署好的环境中

数据库表生成实体类

问题描述 各种框架,不少都是数据库表生成实体类,讨论一下为何要把表对应一个类?直接sql操作数据库表也不错呀 解决方案 解决方案二:引用楼主zhuce2015的回复: 各种框架,不少都是数据库表生成实体类,讨论一下为何要把表对应一个类?直接sql操作数据库表也不错呀 把表生成实体类,可以List<实体类>在程序中使用,这样的话遍历List<>的时候可以通过实体类找到属性名称直接sql填充到DataTable,遍历DataTable的时候,你要么通过列索引.要么通过列名,不如使用实体

Mybatis实体类和表映射问题(推荐)_java

本文是小编给大家带来的mybatis中实体类和表映射问题的知识,学习本教程能够快速帮助我们解决字段名与实体类属性名不相同的冲突问题,需要的朋友一起看看吧! 一.准备演示需要使用的表和数据 CREATE TABLE orders( order_id INT PRIMARY KEY AUTO_INCREMENT, order_no VARCHAR(20), order_price FLOAT ); INSERT INTO orders(order_no, order_price) VALUES('a

【实体类变形】—— 元数据(另类ORM) 描述字段的数据

       放假了,不知道有没有加班的,先祝大家国庆节快乐!      上次说得有点乱,"行列转换"这个词可能误导了大家,那么把这个词扔掉吧.我们重新开始.假设我们有一个News表,我们要往里面添加数据,我们先只考虑保存数据的部分. 一.我们定义一个类.变形的"实体类"  public class ColumnsInfoBase            {         #region 字段的基本信息的描述         /// <summary>

简单实体类和xml文件的相互转换方法

最近写一个题目,要求将一组员工实体类转换成xml文件,或将xml文件转换成一组实体类.题目不难,但写完感觉可以利用泛型和反射将任意一个实体类和xml文件进行转换.于是今天下午立马动手 试了下,做了个简单的模型,可以将简单的实体类和xml文件进行相互转换,但对实体类的属性类型有限制,目前只支持String, Integer, Double三种类型.但是后面可以扩展. 我的大概思路是这样的,只要能拿到实体类的类型信息,我就能拿到实体类的全部字段名称和类型,拼属性的set和get方法更是简单明了,这时

orm-关于hibernate实体类ID自增长的问题

问题描述 关于hibernate实体类ID自增长的问题 一个Java web项目,实体ID是这样的:BEIJING00000001 自增长的 用的ORM是hibernate. 现在用的自增长只能这么增长:1 ,2,3,4... 不能自己补上的前面的部分. 必须在服务层给补上或去掉,很满帆,不知道hibernate有解决这个问题更简单的方案吗? 解决方案 可以考虑Hibernate主键的 自定义主键生成策略. 配置如下: 写一个com.XXX.XXX.MYIDGenerator实现id策略接口Id