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

Posted 韩曙亮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了GroovyMOP 元对象协议与元编程 ( 使用 Groovy 元编程进行函数拦截 | 重写 MetaClass#invokeMethod 方法实现函数拦截 | 实现函数调用转发 )相关的知识,希望对你有一定的参考价值。

文章目录





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



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

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

class Student

    def name;

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


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

覆盖 student.metaClass 的 invokeMethod 方法 ,

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

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





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



使用

student.metaClass.invokeMethod = 

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

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


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

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




三、完整代码示例



完整代码示例 :

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()

执行结果 :

invokeMethod : String name : hello , Object args : []
Hello Tom
Hello Jerry

以上是关于GroovyMOP 元对象协议与元编程 ( 使用 Groovy 元编程进行函数拦截 | 重写 MetaClass#invokeMethod 方法实现函数拦截 | 实现函数调用转发 )的主要内容,如果未能解决你的问题,请参考以下文章

GroovyMOP 元对象协议与元编程 ( 方法注入 | 使用 ExpandoMetaClass 进行方法注入 )

GroovyMOP 元对象协议与元编程 ( 方法委托 | 批量方法委托 )

GroovyMOP 元对象协议与元编程 ( 方法合成引入 | 类内部获取 HandleMetaClass )

GroovyMOP 元对象协议与元编程 ( 方法注入 | 使用 MetaClass 进行方法注入构造方法 )

GroovyMOP 元对象协议与元编程 ( 方法注入 | 使用 Category 分类注入方法 )

GroovyMOP 元对象协议与元编程 ( 方法注入 | 使用 MetaClass 注入静态方法 )