MOP——方法拦截

前面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 实例对象

两种拦截方法的使用场景

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

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,896评论 18 139
  • 前面MOP——方法拦截介绍了利用 MOP 对方法的调用进行拦截,接下来要介绍利用 MOP 实现方法的注入。 方法拦...
    zachaxy阅读 541评论 0 0
  • Groovy学习目录-传送门 元编程(Metaprogramming)->百度百科 Groovy语言支持两种类型的...
    化作春泥_阅读 9,093评论 0 19
  • MOP 总结 之前介绍了基于 MOP 技术: MOP——方法拦截 MOP——方法注入 MOP——方法合成 接下来对...
    zachaxy阅读 1,234评论 0 0
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,947评论 6 342