在Java中使用默认值创建Annotation实例

问题描述 投票:33回答:7

如何创建以下注释的实例(所有字段都设置为其默认值)。

    @Retention( RetentionPolicy.RUNTIME )
    public @interface Settings {
            String a() default "AAA";
            String b() default "BBB";
            String c() default "CCC";
    }

我试过new Settings(),但这似乎不起作用......

java annotations instantiation
7个回答
39
投票

要创建实例,您需要创建一个实现以下的类:

  • java.lang.Annotation
  • 和你想要“模拟”的注释

例如:public class MySettings implements Annotation, Settings

但是你需要特别注意根据equals界面正确实现hashCodeAnnotationhttp://download.oracle.com/javase/1.5.0/docs/api/java/lang/annotation/Annotation.html

如果你不想一次又一次地实现它,那么看看javax.enterprise.util.AnnotationLiteral类。这是CDI(Context Dependency Injection)-API的一部分。 (@see code)

要获取默认值,您可以使用Adrian描述的方式。 Settings.class.getMethod("a").getDefaultValue()


35
投票

您无法创建实例,但至少可以获取默认值

Settings.class.getMethod("a").getDefaultValue()
Settings.class.getMethod("b").getDefaultValue()
Settings.class.getMethod("c").getDefaultValue()

然后,可以使用动态代理返回默认值。就我所知,这也是Java本身处理注释的方式。

class Defaults implements InvocationHandler {
  public static <A extends Annotation> A of(Class<A> annotation) {
    return (A) Proxy.newProxyInstance(annotation.getClassLoader(),
        new Class[] {annotation}, new Defaults());
  }
  public Object invoke(Object proxy, Method method, Object[] args)
      throws Throwable {
    return method.getDefaultValue();
  }
}

Settings s = Defaults.of(Settings.class);
System.out.printf("%s\n%s\n%s\n", s.a(), s.b(), s.c());

24
投票

我编译并在下面运行,结果令人满意。

class GetSettings {
    public static void main (String[] args){
      @Settings final class c { }
      Settings settings = c.class.getAnnotation(Settings.class);
      System.out.println(settings.aaa());
    }
}

3
投票

有同样的问题,我解决了如下。

public static FieldGroup getDefaultFieldGroup() {
    @FieldGroup
    class settring {
    }
    return settring.class.getAnnotation(FieldGroup.class);
}

1
投票

如果与方法一起使用:

@Settings
public void myMethod() {
}

现在,您的注释使用默认值进行初始化。


0
投票

这适用于Sun / Oracle Java 5,6,7,8 :(但由于涉及sun类,可能会破坏Java 9)。 //编辑刚刚确认这仍然适用于OpenJDK 9b59。

package demo;

import sun.reflect.annotation.AnnotationParser;

import java.lang.annotation.*;
import java.lang.reflect.Method;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;

public class AnnotationProxyExample
{

    public static void main(String[] args)
    {

        System.out.printf("Custom annotation creation: %s%n", 
                createAnnotationInstance(Collections.singletonMap("value", "required"), Example.class));

        System.out.printf("Traditional annotation creation: %s%n", 
                X.class.getAnnotation(Example.class));
    }

    private static <A extends Annotation> A createAnnotationInstance(Map<String, Object> customValues, Class<A> annotationType)
    {

        Map<String, Object> values = new HashMap<>();

        //Extract default values from annotation
        for (Method method : annotationType.getDeclaredMethods())
        {
            values.put(method.getName(), method.getDefaultValue());
        }

        //Populate required values
        values.putAll(customValues);

        return (A) AnnotationParser.annotationForMap(annotationType, values);
    }

    @Example("required")
    static class X
    {
    }

    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.TYPE)
    @interface Example
    {
        String value();
        int foo() default 42;
        boolean bar() default true;
    }
}

输出:

Custom annotation creation: @demo.AnnotationProxyExample$Example(bar=true, foo=42, value=required)
Traditional annotation creation: @demo.AnnotationProxyExample$Example(bar=true, foo=42, value=required)

0
投票

如果你有能力改变Settings类的主体,还有另一种解决方案:

@Retention( RetentionPolicy.RUNTIME )
public @interface Settings {
        String DEFAULT_A = "AAA";
        String DEFAULT_B = "BBB";
        String DEFAULT_C = "CCC";

        String a() default DEFAULT_A;
        String b() default DEFAULT_B;
        String c() default DEFAULT_C;
}

然后你可以简单地引用Settings.DEFAULT_A(是的,更好的名字会有所帮助!)。

© www.soinside.com 2019 - 2024. All rights reserved.