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

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

文章目录

  • 一、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 脚本的字节码文件内容 :
代码语言:javascript复制
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 类字节码文件内容 :
代码语言:javascript复制
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 ;

代码语言:javascript复制
        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 对象进行方法注入 ;

0 人点赞