【Groovy】MOP 元对象协议与元编程 ( 方法合成 | 动态注入方法 )

2023-03-30 11:02:02 浏览数 (2)

文章目录

  • 一、动态注入方法
  • 二、完整代码示例

一、动态注入方法


调用 Student 类不存在的方法 , 如果该类重写了

代码语言:javascript复制
def methodMissing(String name, def args)

方法 , 就会回调该函数 , 并且可以从参数中拿到方法名和参数列表 ;

在 methodMissing 方法中 , 可以动态注入该不存在的函数 ;

首先 , 获取 org.codehaus.groovy.runtime.HandleMetaClass 类 , 先将 this 赋值给 Student 对象 , 然后通过 Student 对象获取 metaClass ;

代码语言:javascript复制
        // 先将 this 赋值给 Student 对象
        // 然后通过 Student 对象获取 metaClass
        Student student = this
        println student.metaClass

然后 , 根据方法名称 , 动态注入方法 ; 使用 student.metaClass."方法名" = {闭包} 代码进行方法注入 , 注册前 , 不知道方法名称 , 运行时动态确定注入的方法名 ;

代码语言:javascript复制
        println "动态注入 ${name} 方法, 开始注入!"

        // 动态注入方法
        student.metaClass."${name}" = {
            println "执行动态注入 ${name} 方法, 执行相关操作!"
        }

        println "动态注入 ${name} 方法, 注入完毕!"

最后 , 方法注入之后 , 使用 "方法名"(参数列表) 代码调用注入的方法 , 只需要知道方法名就可以调用该方法 ;

代码语言:javascript复制
        // 调用上述动态注入的方法
        // 注意这里传入的参数, 可以直接传入闭包中
        "$name"(args)

二、完整代码示例


完整代码示例 :

代码语言:javascript复制
class Student {
    def methodMissing(String name, def args) {
        // 直接获取 metaClass
        println metaClass

        // 先将 this 赋值给 Student 对象
        // 然后通过 Student 对象获取 metaClass
        Student student = this
        println student.metaClass

        println "动态注入 ${name} 方法, 开始注入!"

        // 动态注入方法
        student.metaClass."$name" = {
            println "执行动态注入 ${name} 方法, 执行相关操作!"
        }

        println "动态注入 ${name} 方法, 注入完毕!"

        // 调用上述动态注入的方法
        // 注意这里传入的参数, 可以直接传入闭包中
        "$name"(args)

        return null
    }
}

def student = new Student()

// 第一次调用 hello 方法 , 方法没有注入 , 先注入再执行
student.hello()
// 第二次调用hello 方法 , 方法之前注入过了 , 可以直接调用
student.hello()

执行结果 :

代码语言:javascript复制
第一次调用 :
groovy.lang.MetaClassImpl@3e3047e6[class Student]
org.codehaus.groovy.runtime.HandleMetaClass@3e3047e6[groovy.lang.MetaClassImpl@3e3047e6[class Student]]
动态注入 hello 方法, 开始注入!
动态注入 hello 方法, 注入完毕!
执行动态注入 hello 方法, 执行相关操作!
第二次调用 :
执行动态注入 hello 方法, 执行相关操作!

0 人点赞