当前位置 博文首页 > JavaEdge全是干货的技术号:【BAT面试题典】子类可以继承到父类

    JavaEdge全是干货的技术号:【BAT面试题典】子类可以继承到父类

    作者:[db:作者] 时间:2021-07-07 12:44

    子类重写父类方法后,可以继承方法上的注解吗?


    这个不急,让我来分析一下,假设有如下注解:

    • 定义被注解的类

    • 子类直接继承父类

    • 获取父子类和方法的注解信息,并输出注解的value属性的值

    • 日志输出

    可见子类及子类的方法,无法自动继承父类和父类方法上的注解

    不对呀,你得使用@Inherited元注解才能实现注解的继承!行,那咱就加上

    • 再看一遍控制台信息

      可见使用@Inherited只能实现类上的注解继承。

    那么如何实现方法上注解的继承呢?

    最简单暴力地,可通过反射技术,在继承链找到对应方法上的注解。但这样很麻烦,还需要考虑桥接方法。幸好Spring足够强大,提供了AnnotatedElementUtils类。

    @Inherited的支持

    遵循get语义的方法将遵循Java的@Inherited注解的约定,除了在本地声明的批注(包括自定义组成的注解)优于继承的注解之外。相反,遵循find语义的方法将完全忽略@Inherited的存在,因为find搜索算法手动遍历类型和方法层次结构,从而隐式支持注解继承,而无需@Inherited

    Find V.S Get Semantics

    此类中的方法使用的搜索算法遵循find或get语义。

    Get 语义

    仅限于搜索存在于AnnotatedElement上的注解(即在本地声明或继承)或在AnnotatedElement上方的注解层次结构中声明的注释。

    Find 语义

    更加详尽,提供了获取语义以及对以下内容的支持:

    • 搜索接口(如果带注释的元素是类)
    • 搜索超类(如果带注释的元素是一个类)
    • 解析桥接方法(如果带注释的元素是方法)
    • 如果带注解的元素是方法,则在接口中搜索方法
    • 如果带注解的元素是方法,则在超类中搜索方法

    如下俩方法其实也很相像,有何区别呢?

    findAllMergedAnnotations

    Find 对应 SearchStrategy.TYPE_HIERARCHY

    findMergedAnnotation方法可一次性找出父类和接口、父类方法和接口方法上的注解

    getAllMergedAnnotations

    Get对应 SearchStrategy.INHERITED_ANNOTATIONS

    想想 Spring 的@Service、@Controller 等注解支持继承吗?

    我们通常的controller类,都会使用controller注解,如果可以被继承的话,Spring就不会只让我们使用Controller注解了,会提供另一种方式注入Controller组件,就是继承BaseController类。
    Spring 官方对此也有回应:继承的问题在于那些注解真的应该应用于特定的具体类。

    参考

    • https://github.com/spring-projects/spring-framework/issues/8859
    • https://docs.oracle.com/javase/8/docs/api/java/lang/annotation/Inherited.html
      https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/core/annotation/AnnotatedElementUtils.html
      https://docs.oracle.com/javase/8/docs/api/java/lang/Class.html
    cs