获取Haxe宏中的对象类型

mon*_*ero 5 macros haxe

我想在宏中获取一个对象的类,以便我可以访问它的静态变量:

// autoBuild macro adds static field "id_ : Int" to all subclasses
class Base {

}

class Child1 extends Base {
  public function new() {}
}

class Child2 extends Base {
  public function new() {}
}

class Container {
  public function addChild(index: Int, object: Base) {}

  macro function add(object: ???) {
    // find out class of on object
    // ???
    // var id = class.id_;
    this.addChild(id, object);
  }
}
Run Code Online (Sandbox Code Playgroud)

所需用法:

var c = new Container();
c.add(new Child1());
c.add(new Child2());
Run Code Online (Sandbox Code Playgroud)

Gam*_*a11 3

您可以使用Context.typeof()来获取表达式的类型 - 然后您需要进行一些模式匹配来找出类型的名称。以下仅适用于类,因为它仅匹配TInst,但可以扩展:

import haxe.macro.Context;
import haxe.macro.Expr;

class Container {
    // [...]

    public macro function add(self:Expr, object:Expr):Expr {
        var name = switch (Context.typeof(object)) {
            case TInst(_.get() => t, _): t.name;
            case _: throw "object type not found";
        }
        return macro $self.addChild($i{name}.id_, $object);
    }
}
Run Code Online (Sandbox Code Playgroud)

这将生成以下代码:

var c = new Container();
c.addChild(Child1.id_, new Child1());
c.addChild(Child2.id_, new Child2());
Run Code Online (Sandbox Code Playgroud)

请注意,_id只有在实际导入(或顶级)时,通过其非限定名称进行访问才是安全的 - 实际上,您需要t.pack结合使用$p{}来生成完全限定的路径。