美文网首页程序员
MOP——方法拦截

MOP——方法拦截

作者: zachaxy | 来源:发表于2017-12-09 17:21 被阅读0次

前面Groovy 对象和 MOP简单了介绍了 Groovy 所提供的 MOP 机制。
接下来要介绍利用 MOP 实现方法的拦截。
拦截:在对方法进行调用时,我们可能有一些其它的要求,eg:日志的记录,执行时间的记录等,在 Java 可以 AOP 对方法的调用进行拦截,插入我们想执行的方法。而 Groovy 提供的 MOP 机制可以更为轻松的实现对方法的拦截。

MOP 的拦截有两种实现方式:

  1. 实现 GroovyInterceptable 接口进行拦截
  2. 利用 MetaClass 进行拦截

实现 GroovyInterceptable 接口进行拦截

对于一个想要拦截方法的 Groovy 对象,要先实现 GroovyInterceptable 接口,然后务必重写其中的 invokeMethod()方法。我们拦截处理的逻辑都在这个方法中实现。
接下来在调用该类的任何方法的时候,都会被路由到 invokeMethod 方法中。我们可以在 invokeMethod 方法中,根据被拦截的方法名,来加入我们要记录的内容,然后将方法在路由到具体实际的方法中。

class Man implements GroovyInterceptable {
    def talk() {
        System.out.println "hello"
    }

    def run() {
        System.out.println "running"
    }

    def invokeMethod(String name, args) {
        System.out.println("start call method: " + name);
        def targetMethod = Man.metaClass.getMetaMethod(name, args)
        targetMethod.invoke(this,args)
        System.out.println("end call method: " + name);

    }
}

man = new Man()

man.talk()
man.run()

结果:

start call method: talk
hello
end call method: talk
start call method: run
running
end call method: run

我们发现,man 调用的所有方法都被拦截到了 invokeMethod 方法中。

这里要注意的有两点:

  1. 不仅我们调用的方法会被拦截,方法内使用的 Groovy 上的方法(eg:println()方法)同样也会被拦截。所以这里使用了System.out.println()方法,因为其是 Java 上的方法,不会受到拦截的影响,我们这个例子很简单,如果拦截的方法内部使用了 Groovy 中的方法,那么很可能会产生递归调用,导致栈溢出。
  2. 在 invokeMethod 方法中,利用类似反射的手段,根据方法名找到对应的方法,其返回值在 Groovy 中是 MetaMethod,其类型和 Java 中的 Method 类型相似,这里用 targetMethod 表示返回值,但是这里必须显示的使用 def 定义 targetMethod,否则报错groovy.lang.MissingPropertyException: No such property: targetMethod for class: Man 原因可参考深入理解 Android(一):Gradle 详解脚本中的变量和作用域一节的讲解。

额(⊙o⊙)…坑好多

利用 MetaClass 进行拦截

使用 MetaClass 重写上述的方法

class Man {
    def talk() {
        System.out.println "hello"
    }

    def run() {
        System.out.println "running"
    }
}


Man.metaClass.invokeMethod = {
    String name, args ->
        System.out.println("start call method: " + name);
        targetMethod = Man.metaClass.getMetaMethod(name, args)
        targetMethod.invoke(delegate, args)
        System.out.println("end call method: " + name);
}

man = new Man()

man.talk()
man.run()

其结果和实现 GroovyInterceptable 接口的结果相同

这里要注意的是:

  1. 这里将一个闭包赋给了 Man.metaClass 的 invokeMethod,也就是间接的实现了 invokeMethod 方法。因为 groovy 中调用方法时,如果找不到对应的方法名,会从该类的闭包中查找是否有相同的名字,若该闭包的参数也符合,那么就会直接调用该闭包。如果你一时接受不了这种方式,你可以重新创建一个 MetaClass 的实例,重写 invokeMethod 方法,然后将自己实现 metaClass 赋给 Man.metaClass,两种方式是等价的,但是明显前者的方式更简洁。
  2. 实现的闭包中一定注意其有两个参数。否则依然无法实现拦截
  3. 在闭包中,不用为 targetMethod 加 def 的定义
  4. 执行方法时,调用 invoke 方法,第一个参数要传 delegate。因为这是在闭包中。

所有形如 Foo.metaClass.bar = { delegate } 的闭包中出现的 delegate,该 delegate 代表的就是调用 bar 方法的 Foo 实例对象

两种拦截方法的使用场景

第一种需要实习接口,这就需要我们可以修改该类的源码才能让该类实现接口,否则只能使用第二种方法。

相关文章

  • MOP总结

    MOP 总结 之前介绍了基于 MOP 技术: MOP——方法拦截 MOP——方法注入 MOP——方法合成 接下来对...

  • MOP——方法注入

    前面MOP——方法拦截介绍了利用 MOP 对方法的调用进行拦截,接下来要介绍利用 MOP 实现方法的注入。 方法拦...

  • MOP——方法拦截

    前面Groovy 对象和 MOP简单了介绍了 Groovy 所提供的 MOP 机制。接下来要介绍利用 MOP 实现...

  • Groovy——方法合成

    前面MOP——方法注入介绍了利用 MOP 对方法的调用进行注入,接下来要介绍利用 MOP 实现方法的合成。 合成:...

  • 011-Opencv笔记-开操作-闭操作-形态学梯度-顶帽-黑帽

    API morphologyEx(src, dest, CV_MOP_GRADIENT); 开操作 CV_MOP_...

  • StreamNative 宣布开源 MoP:Apache Pul

    我们很高兴地宣布 StreamNative 开源了 “MoP”(MQTT on Pulsar)。MoP 将 MQT...

  • 拦截器Interceptor

    拦截器怎么写 拦截器配置可以从struts-default.xml中找到相关资源 拦截器拦截制定方法拦截指定的方法...

  • 登录拦截器

    定义注解和拦截器 自定义注解,对标记了注解的方法进行拦截 定义拦截器,根据注解进行方法拦截 配置拦截器 使用

  • SpringMVC拦截器

    SpringMVC拦截器;全局拦截和局部拦截 在拦截类中实现HandlerInterceptor,中3个抽象方法 ...

  • Cordova方法

    一,Js与oc交互: 方法1,拦截请求;方法2,拦截页面跳转 方法1,js使用 XMLHttpRequest 发起...

网友评论

    本文标题:MOP——方法拦截

    本文链接:https://www.haomeiwen.com/subject/fyhpixtx.html