Is there something like Annotation Inheritance in java?

前端 未结 4 1985
走了就别回头了
走了就别回头了 2020-11-27 03:21

I\'m exploring annotations and came to a point where some annotations seems to have a hierarchy among them.

I\'m using annotations to generate code in the backgroun

相关标签:
4条回答
  • 2020-11-27 03:40

    Check out https://github.com/blindpirate/annotation-magic , which is a library I developed when I had the same question.

    @interface Animal {
        boolean fluffy() default false;
    
        String name() default "";
    }
    
    @Extends(Animal.class)
    @Animal(fluffy = true)
    @interface Pet {
        String name();
    }
    
    @Extends(Pet.class)
    @interface Cat {
        @AliasFor("name")
        String value();
    }
    
    @Extends(Pet.class)
    @interface Dog {
        String name();
    }
    
    @interface Rat {
        @AliasFor(target = Animal.class, value = "name")
        String value();
    }
    
    @Cat("Tom")
    class MyClass {
        @Dog(name = "Spike")
        @Rat("Jerry")
        public void foo() {
        }
    }
    
            Pet petAnnotation = AnnotationMagic.getOneAnnotationOnClassOrNull(MyClass.class, Pet.class);
            assertEquals("Tom", petAnnotation.name());
            assertTrue(AnnotationMagic.instanceOf(petAnnotation, Animal.class));
    
            Animal animalAnnotation = AnnotationMagic.getOneAnnotationOnClassOrNull(MyClass.class, Animal.class);
            assertTrue(animalAnnotation.fluffy());
    
            Method fooMethod = MyClass.class.getMethod("foo");
            List<Animal> animalAnnotations = AnnotationMagic.getAnnotationsOnMethod(fooMethod, Animal.class);
            assertEquals(Arrays.asList("Spike", "Jerry"), animalAnnotations.stream().map(Animal::name).collect(toList()));
    
    0 讨论(0)
  • 2020-11-27 03:51

    You can annotate your annotation with a base annotation instead of inheritance. This is used in Spring framework.

    To give an example

    @Target(value = {ElementType.ANNOTATION_TYPE})
    public @interface Vehicle {
    }
    
    @Target(value = {ElementType.TYPE})
    @Vehicle
    public @interface Car {
    }
    
    @Car
    class Foo {
    }
    

    You can then check if a class is annotated with Vehicle using Spring's AnnotationUtils:

    Vehicle vehicleAnnotation = AnnotationUtils.findAnnotation (Foo.class, Vehicle.class);
    boolean isAnnotated = vehicleAnnotation != null;
    

    This method is implemented as:

    public static <A extends Annotation> A findAnnotation(Class<?> clazz, Class<A> annotationType) {
        return findAnnotation(clazz, annotationType, new HashSet<Annotation>());
    }
    
    @SuppressWarnings("unchecked")
    private static <A extends Annotation> A findAnnotation(Class<?> clazz, Class<A> annotationType, Set<Annotation> visited) {
        try {
            Annotation[] anns = clazz.getDeclaredAnnotations();
            for (Annotation ann : anns) {
                if (ann.annotationType() == annotationType) {
                    return (A) ann;
                }
            }
            for (Annotation ann : anns) {
                if (!isInJavaLangAnnotationPackage(ann) && visited.add(ann)) {
                    A annotation = findAnnotation(ann.annotationType(), annotationType, visited);
                    if (annotation != null) {
                        return annotation;
                    }
                }
            }
        }
        catch (Exception ex) {
            handleIntrospectionFailure(clazz, ex);
            return null;
        }
    
        for (Class<?> ifc : clazz.getInterfaces()) {
            A annotation = findAnnotation(ifc, annotationType, visited);
            if (annotation != null) {
                return annotation;
            }
        }
    
        Class<?> superclass = clazz.getSuperclass();
        if (superclass == null || Object.class == superclass) {
            return null;
        }
        return findAnnotation(superclass, annotationType, visited);
    }
    

    AnnotationUtils also contains additional methods for searching for annotations on methods and other annotated elements. The Spring class is also powerful enough to search through bridged methods, proxies, and other corner-cases, particularly those encountered in Spring.

    0 讨论(0)
  • 2020-11-27 03:53

    Unfortunately, no. Apparently it has something to do with programs that read the annotations on a class without loading them all the way. See Why is it not possible to extend annotations in Java?

    However, types do inherit the annotations of their superclass if those annotations are @Inherited.

    Also, unless you need those methods to interact, you could just stack the annotations on your class:

    @Move
    @Page
    public class myAwesomeClass {}
    

    Is there some reason that wouldn't work for you?

    0 讨论(0)
  • 2020-11-27 04:06

    In addition to Grygoriys answer of annotating annotations.

    You can check e.g. methods for containing a @Qualifier annotation (or an annotation annotated with @Qualifier) by this loop:

    for (Annotation a : method.getAnnotations()) {
        if (a.annotationType().isAnnotationPresent(Qualifier.class)) {
            System.out.println("found @Qualifier annotation");//found annotation having Qualifier annotation itself
        }
    }
    

    What you're basically doing, is to get all annotations present on the method and of those annotations you get their types and check those types if they're annotated with @Qualifier. Your annotation needs to be Target.Annotation_type enabled as well to get this working.

    0 讨论(0)
提交回复
热议问题