【Groovy】MOP 元对象协议与元编程 ( Groovy 类内部和外部分别获取 metaClass | 分析获取 metaClass 操作的字节码 | HandleMetaClass 注入方法 )

一、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 对象进行方法注入 ;


上一篇:【Android Gradle】安卓应用构建流程 ( Java 源码编译 和 AIDL 文件编译 )(二)


下一篇:【Android Gradle】安卓应用构建流程 ( Java 源码编译 和 AIDL 文件编译 )(一)