理解Java注解和自定义注解

1、元注解:

1)元注解的作用:
就是负责注解, 定义为注解的注解。

2)4个标准的元注解:
Java5.0定义了4个标准的meta-annotation类型,它们被用来提供对其它 annotation类型作说明。Java5.0定义的元注解:

1.@Target,
2.@Retention,
3.@Documented,
4.@Inherited

上述四个元注解位于: java.lang.annotation 包中。

2、@Target 详解

1)概述
@Target说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如切面拦截的方法参数)
作用:用于描述注解的使用范围(即:被描述的注解可以用在什么地方)

取值(ElementType)有 :
1. CONSTRUCTOR : 用于描述构造器
2. FIELD : 用于描述域(属性)
3. LOCAL_VARIABLE : 用于描述局部变量
4. METHOD : 用于描述方法
5. PACKAGE : 用于描述包
6. PARAMETER : 用于描述参数
7. TYPE : 用于描述类、接口(包括注解类型) 或enum声明

2)案例
定义

@Target(ElementType.TYPE)
public @interface Table {
    public String tableName() default "className";
}

@Target(ElementType.FIELD)
public @interface NoDBColumn {
}

@Target({ElementType.TYPE,ElementType.FIELD,ElementType.PARAMETER})
public @interface CustomAnnotation {}

3、@Retention详解

1)概述
@Retention定义了该Annotation被保留的时间长短:
某些Annotation仅出现在源代码中,而被编译器丢弃;
而另一些却被编译在class文件中;
编译在class文件中的Annotation可能会被虚拟机忽略,而另一些在class被装载时将被读取(请注意并不影响class的执行,因为Annotation与class在使用上是被分离的)。使用这个meta-Annotation可以对 Annotation的“生命周期”限制。
作用:用于描述注解的生命周期(即:被描述的注解在什么范围内有效)

取值(RetentionPoicy)有:
    1. SOURCE : 在源文件中有效(即源文件保留)
    2. CLASS : 在class文件中有效(即class保留)
    3. RUNTIME : 在运行时有效(即运行时保留)

2)案例
定义

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.FIELD})
public @interface CustomAnnotation {
    /**
     * 是否为空
     * @return
     */
    boolean isNull() default false;

    /**
     * 最大长度
     * @return
     */
    int maxLength() default 8;

    /**
     * 字段描述
     * @return
     */
    String description() default "";
}

4、@Documented

1)概述
@Documented 用于描述其它类型的annotation应该被作为被标注的程序成员的公共API,因此可以被例如javadoc此类的工具文档化。Documented是一个标记注解,没有成员。
2)案例
定义:

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.FIELD})
@Documented
public @interface CustomAnnotation {
    boolean isNull() default false;
}

5、@Inherited 详解

1)概述
  @Inherited 元注解是一个标记注解,@Inherited阐述了某个被标注的类型是被继承的。如果一个使用了@Inherited修饰的annotation类型被用于一个class,则这个annotation将被用于该class的子类。

  注意:@Inherited annotation类型是被标注过的class的子类所继承。类并不从它所实现的接口继承annotation,方法并不从它所重载的方法继承annotation。

  当@Inherited annotation类型标注的annotation的Retention是RetentionPolicy.RUNTIME,则反射API增强了这种继承性。如果我们使用java.lang.reflect去查询一个@Inherited annotation类型的annotation时,反射代码检查将展开工作:检查class和其父类,直到发现指定的annotation类型被发现,或者到达类继承结构的顶层。

2)案例

@Inherited
public @interface CustomAnnotation {
    boolean isNull() default false;
}

6、自定义注解:

1) 概述
 使用@interface自定义注解时,自动继承了java.lang.annotation.Annotation接口,由编译程序自动完成其他细节。在定义注解时,不能继承其他的注解或接口。@interface用来声明一个注解,其中的每一个方法实际上是声明了一个配置参数。方法的名称就是参数的名称,返回值类型就是参数的类型(返回值类型只能是基本类型、Class、String、enum)。可以通过default来声明参数的默认值。

2)定义注解格式:
public @interface 注解名 {定义体}

注解参数的可支持数据类型:
    1. 所有基本数据类型(int,float,boolean,byte,double,char,long,short)
    2. String类型
    3. Class类型
    4. enum类型
    5. Annotation类型
    6. 以上所有类型的数组

  Annotation类型里面的参数该怎么设定:
  第一, 只能用public或默认(default)这两个访问权修饰.例如,String value();这里把方法设为defaul默认类型;   
  第二, 参数成员只能用基本类型byte,short,char,int,long,float,double,boolean八种基本数据类型和 String,Enum,Class,annotations等数据类型,以及这一些类型的数组.例如,String value();这里的参数成员就为String;  
  第三, 如果只有一个参数成员, 最好把参数名称设为”value”,后加小括号.例:下面的例子FruitName注解就只有一个参数成员。

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface FruitName {
    String value() default "";
}

3)完整注解案例:
定义 :

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.FIELD})
public @interface CustomAnnotation {

    /**
     * 是否为空
     * @return
     */
    boolean isNull() default false;

    /**
     * 最大长度
     * @return
     */
    int maxLength() default 8;

    /**
     * 字段描述
     * @return
     */
    String description() default "";

}

使用:

public class UserInfoRequest {

    @CustomAnnotation(isNull=false,maxLength=4,description="姓名")
    private String name;

    @CustomAnnotation(isNull=false,maxLength=11,description="手机号")
    private String mobile;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getMobile() {
        return mobile;
    }

    public void setMobile(String mobile) {
        this.mobile = mobile;
    }
}

自定义注解就是为了更好的解决问题, 在下一篇中将继续参考下面的地址完成系列编写和理解。




参考文章:
http://www.cnblogs.com/peida/archive/2013/04/24/3036689.html
https://blog.csdn.net/fuyuwei2015/article/details/74898096

猜你喜欢

转载自blog.csdn.net/zhongzunfa/article/details/81253612