zoukankan      html  css  js  c++  java
  • 注解的那些事儿(二)| 如何自定义注解

    自定义注解是自己写框架的必备技能,使用注解能极大地提升开发效率,因此自定义注解是一个高级开发者必备的技能。

    要自定义注解,首先需要了解一个注解的构成部分。

    一个注解大致可以分为三个部分:注解体、元注解、注解属性

    在在这三个主要组成部分中,注解体指定了注解的名字,而元注解则标记了该注解的使用场景、留存时间等信息,而注解属性则指明该注解拥有的属性。

    文章首发于【博客园-陈树义】,点击跳转到原文《注解的那些事儿(二)| 如何自定义注解》

    注解体

    注解体是最简单的一个组成部分,只需要实例中一样有样学样即可。与接口的声明唯一的不同是在 interface 关键字前多了一个 @ 符号。

    //声明了一个名为sweet的注解体
    @Retention(RetentionPolicy.RUNTIME) 
    public @interface sweet{
    }
    

    元注解

    元注解(meta-annotation)本身也是一个注解,用来标记普通注解的存留时间、使用场景、继承属性、文档生成信息。

    元注解是一个特殊的注解,它是 Java 源码中就自带的注解。在Java 中只有四个元注解,它们分别是:@Target、@Retention、@Documented、@Inherited。

    @Target注解

    Target 注解限定了该注解的使用场景。

    它有下面这些取值:

    • ElementType.ANNOTATION_TYPE 可以给一个注解进行注解
    • ElementType.CONSTRUCTOR 可以给构造方法进行注解
    • ElementType.FIELD 可以给属性进行注解
    • ElementType.LOCAL_VARIABLE 可以给局部变量进行注解
    • ElementType.METHOD 可以给方法进行注解
    • ElementType.PACKAGE 可以给一个包进行注解
    • ElementType.PARAMETER 可以给一个方法内的参数进行注解
    • ElementType.TYPE 可以给一个类型进行注解,比如类、接口、枚举
    @Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE}) 
    public @interface Autowired { 
    	boolean required() default true;
    }
    

    在上面 Autowire的 注解中,其 Target 注解的值为 CONSTRUCTOR、METHOD、PARAMETER、FIELD、ANNOTATION_TYPE 这 5 个值。这表示 Autowired 注解只能在构造方法、方法、方法形参、属性、类型这 5 种场景下使用。

    @Retention注解

    Retention 注解用来标记这个注解的留存时间。

    它其有四个可选值:

    • RetentionPolicy.SOURCE。注解只在源码阶段保留,在编译器进行编译时它将被丢弃忽视。
    • RetentionPolicy.CLASS。注解只被保留到编译进行的时候,它并不会被加载到 JVM 中。
    • RetentionPolicy.RUNTIME。注解可以保留到程序运行的时候,它会被加载进入到 JVM 中,所以在程序运行时可以获取到它们。
    @Retention(RetentionPolicy.RUNTIME) 
    public @interface Autowired { 
    	boolean required() default true;
    }
    

    在上面 Autowire的 注解中,其 Retention 注解的值为 RetentionPolicy.RUNTIME,说明该注解会保留到程序运行的时候。

    文章首发于【博客园-陈树义】,点击跳转到原文《注解的那些事儿(二)| 如何自定义注解》

    @Documented

    @ Documented 注解表示将注解信息写入到 javadoc 文档中。

    在默认情况下,我们的注解信息是不会写入到 Javadoc 文档中的。但如果该注解有 @Documented 标识,那么该注解信息则会写入到 javadoc 文档中。

    例如在下面这个例子中,我们声明了一个 @Spicy 的注解,没有 @Documented 元注解。

    public @interface Spicy {
        String spicyLevel();
    }
    

    声明一个 @Sweet 注解,有 @Documented 元注解。

    @Documented
    public @interface Sweet {
        String sweetLevel();
    }
    

    接下来写一个 SweetDemo 类,类中的 sweetWithDoc 方法使用 @Sweet 注解,spicyWithoutDoc 方法使用 @Spicy 注解。

    public class SweetDemo {
        public static void main(String arg[]) {
            new SweetDemo().sweetWithDoc();
            new SweetDemo().spicyWithoutDoc();
        }
        @Sweet (sweetLevel="Level.05")
        public void sweetWithDoc() {
            System.out.printf("sweet With Doc.");
        }
        @Spicy (spicyLevel="Level.04")
        public void spicyWithoutDoc() {
            System.out.printf("spicy Without Doc.");
        }
    }
    

    最后我们使用 Javadoc 命令去生成对应的 JavaDoc 文档,打开文档你会看到:sweetWithDoc方法上面有一个注解信息,而 spicyWithoutDoc 方法上却没有注解信息。

    这个就是 @Documented 这个元注解的作用。

    @Inherited

    @ Inherited注解标识子类将继承父类的注解属性。

    在下面的例子中,我们声明了一个 Sweet 注解,接着在 Peach 类使用了 @Sweet 注解,但是并没有在 RedPeach 类使用该注解。

    //声明一个Sweet注解,标识甜味。
    @Inherited
    @Retention(RetentionPolicy.RUNTIME)
    @interface Sweet {}
    //桃子有甜味
    @Sweet
    public class Peach {}
    //红色的水蜜桃
    public class RedPeach extends Peach {}
    

    虽然我们没在 RedPeach 类上使用了 @Sweet 注解,但是我们在 Sweet 注解声明中使用了 @Inherited 注解,所以 RedPeach 继承了 Peach 的 @Sweet 注解。

    注解属性

    注解属性类似于类方法的声明,注解属性里有三部分信息,分别是:属性名、数据类型、默认值。

    在 @Autowired 注解中就声明了一个名为 required 的 boolean 类型数据,其默认值是 true。

    public @interface Autowired {
    	boolean required() default true;
    }
    

    需要注意的是,注解中定义的属性,它的数据类型必须是 8 种基本数据类型(byte、short、int、long、float、double、boolean、char)或者是类、接口、注解及它们的数组。

    总结

    一个注解大致可以分为三个部分:注解体、元注解、注解属性。在这三个主要组成部分中:注解体指定了注解的名字、元注解则标记了该注解的使用信息,注解属性指明注解的属性。

    学习注解只要知道这三个部分就够了,至于那些繁杂的属性,就用下面这张图来解决吧。用到的时候翻一翻,查一查,足矣!

    文章首发于【博客园-陈树义】,点击跳转到原文《注解的那些事儿(二)| 如何自定义注解》

  • 相关阅读:
    STL标准库algorithm中remove()函数的一个小注意事项
    关于 mem_fun_ref 和 bind2nd的疑问
    记录昨日程序调不通的解释
    复习几个C++概念:声明与定义、传值与拷贝构造、初始化和赋值
    对stl map 赋值添加元素的疑惑 求解(管理员让这个帖子多见会人吧~~谢谢啦!)
    摘抄书上一个逆序字符串的例子(可根据不同的符号,比如*&,.;来拆分单词)
    “指向指针的引用”小例子:忽然豁然开朗~
    论文 “tracking evolving communities in large linked networks” 中不懂的问题和知识总结
    [wp7软件]wp7~~时间日程 软件大全! 集合贴~~~
    [wp7游戏]wp7~~超级血腥类游戏~~集合贴~~
  • 原文地址:https://www.cnblogs.com/chanshuyi/p/annotation_serial_02_self_define_annotation.html
Copyright © 2011-2022 走看看