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

Posted 韩曙亮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了GroovyMOP 元对象协议与元编程 ( 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 对象进行方法注入 ;

以上是关于GroovyMOP 元对象协议与元编程 ( Groovy 类内部和外部分别获取 metaClass | 分析获取 metaClass 操作的字节码 | HandleMetaClass 注入方法 )的主要内容,如果未能解决你的问题,请参考以下文章

GroovyMOP 元对象协议与元编程 ( 方法注入 | 使用 Category 分类进行方法注入的优缺点 )

GroovyMOP 元对象协议与元编程 ( 方法合成引入 | 类内部获取 HandleMetaClass )

GroovyMOP 元对象协议与元编程 ( 方法注入 | 使用 ExpandoMetaClass 进行方法注入 )

GroovyMOP 元对象协议与元编程 ( 方法注入 | 使用 MetaClass 进行方法注入构造方法 )

GroovyMOP 元对象协议与元编程 ( 方法合成 | 动态注入方法 )

GroovyMOP 元对象协议与元编程 ( 通过 MetaMethod#invoke 执行 Groovy 方法 )