使用 Asm 字节码生成器生成泛型类型的方法(类编写器)

使用 Asm 定义简单的 getter 和 setters 很容易(幸运的是,它甚至在他们的 FAQ 中进行了解释)。但是有一件事没有提到,而且我一直找不到文档,那就是如何使用泛型类型信息实现这些。

实际上,我能够很容易地确定泛型类型信息本身(因为代码将采用现有的字段和/或方法,并且存在完整的泛型类型处理和解析)。我只需要为包含泛型类型的类型生成泛型版本。

我希望这像修改签名Asm ClassWriter/MethodVisitor调用一样简单,但是文档中的一些注释表明这可能并不容易(因为泛型信息存储在与常规信息不同的位置)。

编辑:看起来入口点是“ClassWriter.visitField/Method(....,字符串签名) - 请注意,它的”描述“包含正常的非泛型类信息,但术语”签名“(在JLS中)特指泛型 - 包括类型信息。


答案 1

您可以使用 ASM 的 SignatureWriter 类生成签名。

例如,假设您希望为此方法编写签名:

public <K> void doSomething(K thing)

您可以使用以下代码:

SignatureWriter signature = new SignatureWriter();
signature.visitFormalTypeParameter("K");

// Ensure that <K> extends java.lang.Object
{
    SignatureVisitor classBound = signature.visitClassBound();
    classBound.visitClassType(Type.getInternalName(Object.class));
    classBound.visitEnd();
}

// The parameter uses the <K> type variable
signature.visitParameterType().visitTypeVariable("K");

// The return type uses the void primitive ('V')
signature.visitReturnType().visitBaseType('V');

signature.visitEnd();

String signatureString = signature.toString();

这相当于:

String signatureString = "<K:Ljava/lang/Object;>(TK;)V;"

答案 2

根据我的经验,大多数动态字节码生成库对泛型类型没有很好的支持;但是,擦除的类工作得很好(当然,除非您以后想内省这些类)。


推荐