一、Groovy 类内部和外部分别获取 metaClass
在 Groovy 类 内部 和 外部获取的 metaClass 是不同的 ;
代码示例 :
class Student { def methodMissing(String name, def args) { println metaClass return null } } def student = new Student() println student.metaClass student.hello()
打印结果 :
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 的语句是
var3[0].callCurrent(this, this.metaClass);
此处直接调用 this.metaClass 获取 metaClass , 而这个 this,metaClass 是构造方法中通过 this.$getStaticMetaClass() 获取的 MetaClass ;
MetaClass var2 = this.$getStaticMetaClass(); this.metaClass = var2;
在 Student 类外部 ( Groovy 脚本中 ) 获取 metaClass 的语句是
var1[2].callCurrent(this, var1[3].callGetProperty(student));
此处是通过动态调用节点的方式获取 metaClass , 动态获取 student 对象中的属性 ;
不同的调用方式获取的 metaClass 是不同的 ;
三、使用 HandleMetaClass 注入方法
Student 对象内部获取的 groovy.lang.MetaClassImpl , 是不能进行方法注入的 , 注入方法会报错 ;
如果使用 MetaClass 为 Student 类注入方法 , 必须使用 org.codehaus.groovy.runtime.HandleMetaClass 对象进行方法注入 ;