问题

我一直在使用Guice的AOP拦截一些方法调用。我的类实现了一个接口,我想注释接口方法,以便Guice可以选择正确的方法。即使注释类型使用Inherited注释注释,实现类也不会继承继承的java doc中所述的注释:

另请注意,此元注释仅会导致注释从超类继承;已实现接口上的注释无效。

这可能是什么原因?了解对象类在运行时实现的所有接口并不是一件难事,因此必须有充分的理由支持这一决定。


#1 热门回答(117 赞)

我会说原因是否则会出现多重继承问题。
示例:

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD) @Inherited
public @interface Baz { String value(); }

public interface Foo{
    @Baz("baz") void doStuff();
}

public interface Bar{
    @Baz("phleem") void doStuff();
}

public class Flipp{
    @Baz("flopp") public void doStuff(){}
}

public class MyClass extends Flipp implements Foo, Bar{}

如果我这样做:

MyClass.class.getMethod("doStuff").getAnnotation(Baz.class).value()

结果会是什么? 'baz','phleem'或'flopp'?

因此,接口上的注释很少有用。


#2 热门回答(32 赞)

来自Javadocfor @Inherited:

表示自动继承注释类型。如果注释类型声明中存在Inherited元注释,并且用户在类声明上查询注释类型,并且类声明没有此类型的注释,则将自动查询类的超类以获取注释类型。将重复此过程,直到找到此类型的注释,或者到达类层次结构(对象)的顶部。如果没有超类具有此类型的注释,则查询将指示相关类没有此类注释。请注意,如果使用带注释的类型来注释除类之外的任何内容,则此元注释类型不起作用。另请注意,此元注释仅导致注释从超类继承;已实现接口上的注释无效。

另一方面,JSR 305验证器执行某种继承查找。如果你有类的层次结构:

//Person.java
@Nonnull
 public Integer getAge() {...}

//Student.java (inherits from Person)
@Min(5)
public Integer getAge() {...}

然后,有效验证在Student.getAge()are@Nonnull @Min(5).@Nonnull上有no@Inheritedmeta-annotation。


原文链接