6

对于以下自定义 Java 注解

@CustomAnnotation(clazz=SomeClass.class)
public class MyApplicationCode
{ 
   ... 
}

我基本上希望能够在编译时同时获取 MyApplicationCode 的 Class 对象和 clazz 参数,以确认一些编码约定的一致性(另一个故事)。基本上我希望能够访问注释处理器中的 MyApplicationCode.class 和 Someclass.class 代码。我快到了,但我错过了一些东西。我有

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.SOURCE)
public @interface CustomAnnotation
{
   public Class clazz();
}

然后我有处理器:

public class CustomAnnotationProcessor extends AbstractProcessor
{
    private ProcessingEnvironment processingEnvironment;

    @Override
    public synchronized void init(ProcessingEnvironment processingEnvironment)
    {
        this.processingEnvironment = processingEnvironment;
    }

    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment environment)    
    {
        Set<? extends Element> elements = environment.getElementsAnnotatedWith(ActionCommand.class);

        for(Element e : elements)
        {
            Annotation annotation = e.getAnnotation(CustomAnnotation.class);
            Class clazz = ((CustomAnnotation)annotation).clazz();        

            // How do I get the actual CustomAnnotation clazz?
            // When I try to do clazz.getName() I get the following ERROR:
            // Attempt to access Class object for TypeMirror SomeClass

            // Also, how do I get the Class object for the class that has the annotation within it?
            // In other words, how do I get MyApplicationCode.class?
        }
    }
}

所以我在 process 方法中尝试做的是从下面的原始代码中获取 SomeClass.class 和 MyApplication.class 在编译时进行一些自定义验证。我似乎无法终生弄清楚如何获得这两个值......

@CustomAnnotation(clazz=SomeClass.class)
public class MyApplicationCode

更新:以下帖子有更多细节,而且更接近。但问题是你最终仍然得到一个 TypeMirror 对象,从中提取类对象,它没有解释:http ://blog.retep.org/2009/02/13/getting-class-values -来自注释处理器中的注释/

Update2:您可以通过执行获取 MyApplication.class

String classname = ((TypeElement)e).getQualifiedName().toString();
4

3 回答 3

1

我打算为您指明博客http://blog.retep.org/2009/02/13/getting-class-values-from-annotations-in-an-annotationprocessor/的方向,但看起来像您已经找到那个了。

我看到你想出了如何访问 MyApplication 元素,所以我不会介绍它......

您看到的异常实际上包含其中的注释属性的类型。因此,您可以在捕获异常时引用注释 clazz 值:

public class CustomAnnotationProcessor extends AbstractProcessor
{
    private ProcessingEnvironment processingEnvironment;

    @Override
    public synchronized void init(ProcessingEnvironment processingEnvironment)
    {
        this.processingEnvironment = processingEnvironment;
    }

    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment environment)    
    {
        Set<? extends Element> elements = environment.getElementsAnnotatedWith(ActionCommand.class);

        for(Element e : elements)
        {
            CustomAnnotation annotation = e.getAnnotation(CustomAnnotation.class);    

            TypeMirror clazzType = null;
            try {
                annotation.clazz();
            } catch (MirroredTypeException mte) {
                clazzType = mte.getTypeMirror();
            }

            System.out.println(clazzType); // should print out SomeClass

        }
    }
}

是的,这是一个完全破解的解决方案,我不确定为什么 API 开发人员决定使用注释处理器功能朝这个方向发展。但是,我已经看到很多人(包括我自己)实现了这一点,并且提到的文章也描述了这种技术。这似乎是目前可以接受的解决方案。

在“获取”MyApplicationCode 和 SomeClass 的类值方面,如果它们是正在编译的类,您将无法这样做。但是,您可以使用 Element 和 TypeMirror 表示对您的类执行一些高级验证(方法、字段、类名称、存在的注释等)

于 2012-06-15T14:49:04.173 回答
0

在阅读了这个相关的 SO question之后,我发现了这个关于 Java Annotation Processing Tool (APT) 的优秀页面。这是从 2005 年开始的,所以现在可能不是最好的方法。

APT [...] 是 Java 的注解处理工具。更具体地说,APT 允许您插入代码以在代码编译发生时处理源文件中的注释 - 在此过程中,您可以发出注释、警告和错误。

这仅适用于 Oracle 的 JDK。

于 2012-05-19T09:47:38.060 回答
-1

现在是编译时间。我认为编译器甚至还没有完成源代码的编译。您从 AnnotatedElement 实例中检索此类信息,该实例将为您提供已注释类型的相关信息,但不提供其运行时属性,因为相关类文件尚未由虚拟机加载,因此尚不可用。并且编译器甚至不能保证在 ja​​va 的虚拟机下运行,因此不强制要求能够加载类文件。它的要求只是能够生成任何特定虚拟机都可以读取的字节码。

因此,请检查镜像 Api,有关您已注释的类/方法/字段的任何相关信息,请检查表示该实例的 AnnotatedElement。

顺便说一句:这只是我推理出来的信息,所以它可能不是真正的事实。

于 2012-05-18T23:01:17.590 回答