文章目录
- 一、重写 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 ;
// 方法转发 : 调用 student 对象中的原来的方法
// 注意此处不能使用 metaClass.invokeMethod 方法调用对象中的方法 , 会导致栈溢出
// 这里通过 MetaClass#getMetaMethod 获取方法 , 然后执行
def method = student.metaClass.getMetaMethod(name, args)
- 然后 , 执行该 MetaMethod 方法 , 需要传入 对象 和 参数 ;
// 方法不为空再执行该方法
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