【Groovy】MOP 元对象协议与元编程 ( 使用 Groovy 元编程进行函数拦截 | 重写 MetaClass#invokeMethod 方法实现函数拦截 | 实现函数调用转发 )

2023-03-30 10:39:09 浏览数 (1)

文章目录

  • 一、重写 MetaClass#invokeMethod 方法实现函数拦截
  • 二、在 MetaClass#invokeMethod 方法中调用对象的其它方法
  • 三、完整代码示例

一、重写 MetaClass#invokeMethod 方法实现函数拦截


在 Groovy 中 , 如果覆盖了对象的 MetaClass#invokeMethod 方法 , 那么 , 在执行该对象的任何方法时 , 都会回调该 invokeMethod 方法 ;

给定一个类和该类的实例对象 :

代码语言:javascript复制
class Student{

    def name;

    def hello() {
        System.out.println "Hello ${name}"
    }
}

def student = new Student(name: "Tom")

覆盖 student.metaClass 的 invokeMethod 方法 ,

代码语言:javascript复制
// 如果覆盖了 invokeMethod 方法
// 那么 , 执行该对象的任何方法时 , 都会回调该 invokeMethod 方法
student.metaClass.invokeMethod = {
    String name, Object args ->
        System.out.println "invokeMethod : String name : $name , Object args : $args"
}

调用 student 对象的 hello 方法时 , 就会回调该闭包中的方法 , 即使没有实现 GroovyInterceptable 接口 , 也可以进行函数拦截 ;

二、在 MetaClass#invokeMethod 方法中调用对象的其它方法


使用

代码语言:javascript复制
student.metaClass.invokeMethod = {}

重写了 invokeMethod 方法后 , 拦截函数之后 , 需要将方法传递下去 , 调用真正要调用的方法 ;

注意此处不能使用 student.metaClass.invokeMethod 调用其它方法 , 这样会导致无限循环递归调用 , 导致栈溢出异常 ;

在 MetaClass#invokeMethod 方法中调用对象的其它方法 ,

  • 首先 , 要从 student.metaClass 中根据 方法名 和 方法参数 获取指定的 MetaMethod ;
代码语言:javascript复制
        // 方法转发 : 调用 student 对象中的原来的方法
        // 注意此处不能使用 metaClass.invokeMethod 方法调用对象中的方法 , 会导致栈溢出
        // 这里通过 MetaClass#getMetaMethod 获取方法 , 然后执行
        def method = student.metaClass.getMetaMethod(name, args)
  • 然后 , 执行该 MetaMethod 方法 , 需要传入 对象 和 参数 ;
代码语言:javascript复制
        // 方法不为空再执行该方法
        if (method != null) {
            method.invoke(student, args)
        }

三、完整代码示例


完整代码示例 :

代码语言:javascript复制
class Student{

    def name;

    def hello() {
        System.out.println "Hello ${name}"
    }
}

def student = new Student(name: "Tom")
def student2 = new Student(name: "Jerry")

// 如果覆盖了 invokeMethod 方法
// 那么 , 执行该对象的任何方法时 , 都会回调该 invokeMethod 方法
student.metaClass.invokeMethod = {
    String name, Object args ->
        System.out.println "invokeMethod : String name : $name , Object args : $args"

        // 方法转发 : 调用 student 对象中的原来的方法
        // 注意此处不能使用 metaClass.invokeMethod 方法调用对象中的方法 , 会导致栈溢出
        // 这里通过 MetaClass#getMetaMethod 获取方法 , 然后执行
        def method = student.metaClass.getMetaMethod(name, args)
        // 方法不为空再执行该方法
        if (method != null) {
            method.invoke(student, args)
        }
}

// 直接调用 hello 方法
student.hello()
student2.hello()

执行结果 :

代码语言:javascript复制
invokeMethod : String name : hello , Object args : []
Hello Tom
Hello Jerry

0 人点赞