使用 Java doc API 从 DeclaredType/TypeMirror 获取限定类型

Luk*_*uke 5 java javadoc java-11

我有一个DeclaredType字段,我想获得该字段的完全限定类型(原始类型?)。例如,如果该字段是:

public static Optional<String> foo;
Run Code Online (Sandbox Code Playgroud)

我想得到java.util.Optional

目前我可以通过以下方式获取包名称:

env.getElementUtils().getPackageOf(declaredType.asElement());
Run Code Online (Sandbox Code Playgroud)

然而,我回来了同样的问题我结束了一个我能得到的那种类型参数ListTypeMirror,我不知道怎么弄类型的限定名。

我注意到我可以打电话TypeMirror#toString()并且会返回(对于上述)类似的东西:

java.util.Optional<java.lang.String>
Run Code Online (Sandbox Code Playgroud)

我想我可以把前面的所有东西都剪掉,<但这感觉就像一个黑客。

作为参考,这是我获取该字段的方式:

private VariableElement getFieldWithName(DocletEnvironment environment, TypeElement classDoc, String fieldName) {
    for(VariableElement e : ElementFilter.fieldsIn(environment.getElementUtils().getAllMembers(classDoc))) {
        if(e.getSimpleName().toString().equals(fieldName)) {
            return e;
        }
    }
    return null;
}

TypeElement classElement = env.getElementUtils().getTypeElement(MyClass.class.getCanonicalName());
VariableElement fieldDoc = getFieldWithName(env, classElement, "foo");
DeclaredType declaredType = (DeclaredType) fieldDoc.asType();
Run Code Online (Sandbox Code Playgroud)

小智 2

我测试了这段代码,它可以在 Apache Netbeans 10 和 Jdk11 上运行。使用((DeclaredType) variableElement.asType()).asElement().toString()

            DeclaredType declaredType = (DeclaredType) e.asType();//e is VariableElement from the loop
            fullyQualifiedName = declaredType.asElement().toString();
Run Code Online (Sandbox Code Playgroud)

测试课程:

class MyClass1 {
public static Optional<String> foo;
public static List<String> newList;
public static MyClass2 obj;
public Media media;
public void calculate(Double dl) {
}}

class MyClass2 extends MyClass1{
public static Color cl;
}
Run Code Online (Sandbox Code Playgroud)

以及 Doclet实现

public class TestVarElement implements Doclet {

 public void testFields(DocletEnvironment env) {
    TypeElement typeElement = env.getElementUtils().getTypeElement(MyClass1.class.getCanonicalName());
    System.out.println("Test for 'foo': "+getFieldWithName(env,typeElement,"foo"));
    System.out.println("Test for 'newList': "+getFieldWithName(env,typeElement,"newList"));
    System.out.println("Test for 'obj': "+getFieldWithName(env,typeElement,"obj"));
    System.out.println("Test for 'media': "+getFieldWithName(env,typeElement,"media"));
}

    private String getFieldWithName(DocletEnvironment env, TypeElement classDoc, String fieldName) {
        String fullyQualifiedName = "";
        for (VariableElement e : ElementFilter.fieldsIn(env.getElementUtils().getAllMembers(classDoc))) {
            if (e.getSimpleName().toString().equals(fieldName)) {
                DeclaredType declaredType = (DeclaredType) e.asType(); //The type of the VariableElement
                fullyQualifiedName = declaredType.asElement().toString(); //Get the fqn 
                break;
            }
        }
        return fullyQualifiedName;
    }
   @Override
    public boolean run(DocletEnvironment docEnv) {
        testFields(docEnv);
        return true;
    }
 
    ... Other Overrids

    }
Run Code Online (Sandbox Code Playgroud)

调试/运行程序:

     public class NewClass {
            public static void main(String[] args) {
                ToolProvider javadoc=ToolProvider.findFirst("javadoc").orElseThrow();
                int result=javadoc.run(System.out, System.err, new String[]{"-doclet",TestVarElement.class.getName(),"C:\\Users\\Super3\\Documents\\NetBeansProjects\\Myproject\\src\\pk\\TestVarElement.java"});
      
//The following is for java 8 or older, the implementation is diferent where `start` method is used instead of `run`.
     //Main.execute (new String[]{"-doclet",TestVarElement.class.getName(),"C:\\Users\\Super3\\Documents\\NetBeansProjects\\Myproject\\src\\pk\\TestVarElement.java"});
          }
        }
Run Code Online (Sandbox Code Playgroud)

输出

测试“foo”:java.util.Optional

测试“newList”:java.util.List

测试“obj”:pk.MyClass2

测试“媒体”:javax.print.attribute.standard.Media