Создать экземпляр аннотации с настройками по умолчанию в Java
Как создать экземпляр следующей аннотации (со всеми полями, установленными по умолчанию).
@Retention( RetentionPolicy.RUNTIME )
public @interface Settings {
String a() default "AAA";
String b() default "BBB";
String c() default "CCC";
}
Я пробовал new Settings()
, но это не работает...
Ответы
Ответ 1
Вы не можете создать экземпляр, но по крайней мере получить значения по умолчанию
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());
Ответ 2
Чтобы создать экземпляр, вам нужно создать класс, который реализует:
-
java.lang.Annotation
- и аннотацию, которую вы хотите "смоделировать"
Например: public class MySettings implements Annotation, Settings
Но вам нужно обратить особое внимание на правильную реализацию equals
и hashCode
соответствии с интерфейсом Annotation
. http://download.oracle.com/javase/1.5.0/docs/api/java/lang/annotation/Annotation.html
Если вы не хотите реализовывать это снова и снова, взгляните на класс javax.enterprise.util.AnnotationLiteral. Это является частью CDI (внедрение зависимости от контекста) -API. (см. код)
Чтобы получить значения по умолчанию, вы можете использовать способ, описанный Адрианом. Settings.class.getMethod("a").getDefaultValue()
Ответ 3
Я собираю и бегу ниже с удовлетворительными результатами.
class GetSettings {
public static void main (String[] args){
@Settings final class c { }
Settings settings = c.class.getAnnotation(Settings.class);
System.out.println(settings.aaa());
}
}
Ответ 4
имел ту же проблему, я решил ее следующим образом.
public static FieldGroup getDefaultFieldGroup() {
@FieldGroup
class settring {
}
return settring.class.getAnnotation(FieldGroup.class);
}
Ответ 5
Если используется с методом:
@Settings
public void myMethod() {
}
Теперь ваша аннотация инициализируется значениями по умолчанию.
Ответ 6
Это работает с Sun/Oracle Java 5,6,7,8: (но может потенциально сломаться с Java 9 из-за задействованных классов солнца).
//edit Только что проверено, что это все еще работает с 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)
Ответ 7
Существует альтернативное решение, если вы можете позволить себе изменить тело класса 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
(да, лучшее имя поможет!).