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

相关文章
|
4月前
|
设计模式 算法 Java
AOP跨模块捕获异常遭CGLIB拦截而继续向上抛出异常
最近,在开发过程中,我遇到一个不易察觉的小bug。这个bug并没有直接给出报错信息,使得排查问题的根源变得困难。我希望通过分享这个经验,帮助大家避免重蹈覆辙,以免浪费不必要的时间和精力。为了避免类似的困境,我们应当时刻保持警惕,对开发过程中的每一个细节都进行严格的检查。同时,利用调试工具和日志输出等功能,可以帮助我们更快速地定位和解决问题。此外,定期进行代码审查和测试也是非常必要的,这有助于发现潜在的问题并及时解决。
84 1
|
10月前
AOP拦截规则
AOP拦截规则
|
10月前
|
Java Maven Spring
如何通过自定义注解来实现 Spring AOP,以便更加灵活地控制方法的拦截和增强?
如何通过自定义注解来实现 Spring AOP,以便更加灵活地控制方法的拦截和增强?
62 0
|
SQL Java 数据库连接
Mybatis中sql拦截增强-AOP+interceptor实现分页和排序
基于interceptor可以实现sql的完整打印,除了实现打印之外。其实还可以实现分页和排序,下面的分页和排序基于aop+mybatis的interceptor实现。其本质还是对mappedStament的boundSql进行增强。 下面的项目来源于github,通过这个我们可以很好的学习mybatis中插件interceptor的使用。
740 0
Mybatis中sql拦截增强-AOP+interceptor实现分页和排序
Springboot AOP 拦截 直接返回结果数据
Springboot AOP 拦截 直接返回结果数据
322 0
Springboot AOP 拦截 直接返回结果数据
Springboot AOP Aspect 拦截中 获取HttpServletResponse response
Springboot AOP Aspect 拦截中 获取HttpServletResponse response
266 0
Springboot AOP Aspect 拦截中 获取HttpServletResponse response
|
Java 关系型数据库 MySQL
【SpringBoot 基础系列】接口上注解 AOP 拦截不到场景兼容实例演示
在 Java 的开发过程中,面向接口的编程可能是大家的常态,切面也是各位大佬使用 Spring 时,或多或少会使用的一项基本技能;结果这两个碰到一起,有意思的事情就发生了,接口方法上添加注解,面向注解的切面拦截,居然不生效 这就有点奇怪了啊,最开始遇到这个问题时,表示难以相信;事务注解也挺多是写在接口上的,好像也没有遇到这个问题(难道是也不生效,只是自己没有关注到?) 接下来我们好好瞅瞅,这到底是怎么个情况
482 0
【SpringBoot 基础系列】接口上注解 AOP 拦截不到场景兼容实例演示
|
XML 移动开发 Java
【框架】[Spring]AOP拦截-使用切点:AspectJExpressionPointcut-切点语言
【框架】[Spring]AOP拦截-使用切点:AspectJExpressionPointcut-切点语言
215 0
【框架】[Spring]AOP拦截-使用切点:AspectJExpressionPointcut-切点语言
|
XML Java 数据格式
【框架】[Spring]AOP拦截-三种方式实现自动代理(1)
【框架】[Spring]AOP拦截-三种方式实现自动代理
171 0
【框架】[Spring]AOP拦截-三种方式实现自动代理(1)
|
XML Java 数据格式
【框架】[Spring]XML配置实现AOP拦截-切点:JdkRegexpMethodPointcut
【框架】[Spring]XML配置实现AOP拦截-切点:JdkRegexpMethodPointcut
204 0
【框架】[Spring]XML配置实现AOP拦截-切点:JdkRegexpMethodPointcut