文章目录
- 一、Groovy 类内部和外部分别获取 metaClass
- 二、分析 Groovy 类内部和外部获取 metaClass 操作的字节码
- 三、使用 HandleMetaClass 注入方法
一、Groovy 类内部和外部分别获取 metaClass
在 Groovy 类 内部 和 外部获取的 metaClass 是不同的 ;
代码示例 :
代码语言:javascript复制class Student {
def methodMissing(String name, def args) {
println metaClass
return null
}
}
def student = new Student()
println student.metaClass
student.hello()
打印结果 :
代码语言:javascript复制org.codehaus.groovy.runtime.HandleMetaClass@732d0d24[groovy.lang.MetaClassImpl@732d0d24[class Student]]
groovy.lang.MetaClassImpl@732d0d24[class Student]
在 Groovy 脚本中 , 即 Student 外部 , 获取的 student.metaClass 类型是 org.codehaus.groovy.runtime.HandleMetaClass
;
在 Student 对象内部的 methodMissing 方法中 , 获取的 metaClass 类型是 groovy.lang.MetaClassImpl
;
二、分析 Groovy 类内部和外部获取 metaClass 操作的字节码
下面开始分析字节码文件 :
- Groovy 脚本的字节码文件内容 :
public class Groovy extends Script {
public Groovy() {
CallSite[] var1 = $getCallSiteArray();
super();
}
public Groovy(Binding context) {
CallSite[] var2 = $getCallSiteArray();
super(context);
}
public static void main(String... args) {
CallSite[] var1 = $getCallSiteArray();
var1[0].call(InvokerHelper.class, Groovy.class, args);
}
public Object run() {
CallSite[] var1 = $getCallSiteArray();
Object student = var1[1].callConstructor(Student.class);
var1[2].callCurrent(this, var1[3].callGetProperty(student));
return var1[4].call(student);
}
}
- Student 类字节码文件内容 :
public class Student implements GroovyObject {
@Generated
public Student() {
CallSite[] var1 = $getCallSiteArray();
super();
MetaClass var2 = this.$getStaticMetaClass();
this.metaClass = var2;
}
public Object methodMissing(String name, Object args) {
CallSite[] var3 = $getCallSiteArray();
var3[0].callCurrent(this, this.metaClass);
return null;
}
@Generated
@Internal
public MetaClass getMetaClass() {
MetaClass var10000 = this.metaClass;
if (var10000 != null) {
return var10000;
} else {
this.metaClass = this.$getStaticMetaClass();
return this.metaClass;
}
}
@Generated
@Internal
public void setMetaClass(MetaClass var1) {
this.metaClass = var1;
}
}
在 Student 类内部获取 metaClass 的语句是
代码语言:javascript复制var3[0].callCurrent(this, this.metaClass);
此处直接调用 this.metaClass 获取 metaClass , 而这个 this,metaClass 是构造方法中通过 this.$getStaticMetaClass()
获取的 MetaClass ;
MetaClass var2 = this.$getStaticMetaClass();
this.metaClass = var2;
在 Student 类外部 ( Groovy 脚本中 ) 获取 metaClass 的语句是
代码语言:javascript复制var1[2].callCurrent(this, var1[3].callGetProperty(student));
此处是通过动态调用节点的方式获取 metaClass , 动态获取 student 对象中的属性 ;
不同的调用方式获取的 metaClass 是不同的 ;
三、使用 HandleMetaClass 注入方法
Student 对象内部获取的 groovy.lang.MetaClassImpl , 是不能进行方法注入的 , 注入方法会报错 ;
如果使用 MetaClass 为 Student 类注入方法 , 必须使用 org.codehaus.groovy.runtime.HandleMetaClass 对象进行方法注入 ;