在 ByteBuddy 中做循环类型

Doing Cyclic Types in ByteBuddy

我正在尝试使用 byte-buddy 动态创建循环 类:

class A{
 B b;
}
class B{
  A a;
}

我看过这个 并且我写了这个代码。

public static void main(String[] args) throws Exception {

    final ByteBuddy bb = new ByteBuddy();

    TypeDescription.Generic g1 = TypeDescription.Generic.Builder.rawType(Object.class).build();
    final InstrumentedType typeDescrB = InstrumentedType.Default.of("B", g1, Modifier.PUBLIC);

    final DynamicType.Unloaded<Object> madeA = bb
            .subclass(Object.class)
            .name("A")
            .defineField("theB", typeDescrB, Modifier.PUBLIC).make();

    final DynamicType.Unloaded<Object> madeB = bb.subclass(Object.class)
            .name("B")
            .defineField("theA", madeA.getTypeDescription()).make();

    Class a = madeA.include(madeB)
            .load(Test.class.getClassLoader(), ClassLoadingStrategy.Default.WRAPPER)
            .getLoaded();

    for (Field f : a.getFields()) {
        System.out.println(f + "|" + f.getType().getName());
    }
    System.out.println("====");
    for (Field f : a.getField("theB").getType().getFields()) {
        System.out.println(f + "|" + f.getType().getName());
    }
}

在 运行 代码之后我得到了这个结果

public B A.theB|B
====
Process finished with exit code 0

表示ClassB不包含字段a。有人知道问题出在哪里吗?

我猜你所看到的只是代码中的一个错误。

使用此代码:

final DynamicType.Unloaded<Object> madeB = bb.subclass(Object.class)
    .name("B")
    .defineField("theA", madeA.getTypeDescription()).make();

…您已将字段设置为非 public(请注意,您为其他字段指定了 PUBLIC,但此处未指定)。然后使用此代码:

for (Field f : a.getField("theB").getType().getFields()) {
    System.out.println(f + "|" + f.getType().getName());
}

…您只要求 public 个字段。也许你的意思是 getDeclaredFields()?