zoukankan      html  css  js  c++  java
  • java基础之:java注解

    一:元注解

          元注解的作用就是负责注解其他注解。Java5.0定义了4个标准的meta-annotation类型,它们被用来提供对其它 annotation类型作说明。Java5.0定义的元注解:
      1.@Target,
      2.@Retention,
      3.@Documented,
      4.@Inherited
      这些类型和它们所支持的类在java.lang.annotation包中可以找到。下面我们看一下每个元注解的作用和相应分参数的使用说明。

     @Target:

        @Target说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循 环变量、catch参数)。在Annotation类型的声明中使用了target可更加明晰其修饰的目标。

      作用:用于描述注解的使用范围(即:被描述的注解可以用在什么地方)

      取值(ElementType)有:

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



    @Retention:

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

      作用:表示需要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)

      取值(RetentionPoicy)有:

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



    @Documented:

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



    @Inherited:

     

      @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类型被发现, 或者到达类继承结构的顶层.









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

    三:实例

    《一》类注解

     1 package com.yeepay.thead.test;
     2 
     3 import java.lang.annotation.Documented;
     4 import java.lang.annotation.ElementType;
     5 import java.lang.annotation.Inherited;
     6 import java.lang.annotation.Retention;
     7 import java.lang.annotation.RetentionPolicy;
     8 import java.lang.annotation.Target;
     9 /**
    10  * 
    11  * @author shangxiaofei
    12  *(1)使用@interface自定义注解时,自动继承了java.lang.annotation.Annotation接口,由编译程序自动完成其他细节。
    13  *(2)在定义注解时,不能继承其他的注解或接口。
    14  *(3)@interface用来声明一个注解,其中的每一个方法实际上是声明了一个配置参数。
    15  *(4)方法的名称就是参数的名称,返回值类型就是参数的类型(返回值类型只能是基本类型、Class、String、enum)。
    16  *(5)可以通过default来声明参数的默认值。
    17  *(6)因为没有定义public,所以默认的访问权限为包权限,在定义时没有指定默认值,则使用时必须指定默认值  
    18  */
    19 
    20 
    21 @Target({ElementType.TYPE})
    22 @Retention(RetentionPolicy.RUNTIME)
    23 @Documented
    24 @Inherited
    25 public @interface MyAnnotationByClass {
    26     //定义一个name属性
    27     public String name() default "sxf"; 
    28     
    29     //定义一个内部枚举类
    30     public enum SEX{MAN,WOMEN};
    31     
    32     //定义一个性别的属性
    33     public SEX xingbie();
    34     
    35     //定义一个年龄的类型
    36     public int age();
    37 }
    View Code

    《二》属性注解

     1 package com.yeepay.thead.test;
     2 
     3 import java.lang.annotation.Documented;
     4 import java.lang.annotation.ElementType;
     5 import java.lang.annotation.Inherited;
     6 import java.lang.annotation.Retention;
     7 import java.lang.annotation.RetentionPolicy;
     8 import java.lang.annotation.Target;
     9 
    10 @Target(ElementType.FIELD)
    11 @Retention(RetentionPolicy.RUNTIME)
    12 @Documented
    13 @Inherited
    14 public @interface MyAnnotationByField {
    15     
    16     public String value() default "";
    17 }
    View Code

    《三》方法注解

     1 package com.yeepay.thead.test;
     2 
     3 import java.lang.annotation.Documented;
     4 import java.lang.annotation.ElementType;
     5 import java.lang.annotation.Inherited;
     6 import java.lang.annotation.Retention;
     7 import java.lang.annotation.RetentionPolicy;
     8 import java.lang.annotation.Target;
     9 
    10 /**
    11  * 方法注解
    12  * @author shangxiaofei
    13  *
    14  */
    15 @Target({ElementType.METHOD})
    16 @Retention(RetentionPolicy.RUNTIME)
    17 @Documented
    18 @Inherited
    19 public @interface MyAnotationByMethod {
    20     
    21     public String fundcheck() default "";
    22 
    23 }
    View Code

    《四》测试

     1 package com.yeepay.thead.test;
     2 
     3 import java.lang.reflect.Field;
     4 import java.lang.reflect.Method;
     5 
     6 import com.yeepay.thead.test.MyAnnotationByClass.SEX;
     7 
     8 
     9 public class AnotationTest {
    10 
    11     public static void main(String[] args) {
    12         User user=new User("sxf");
    13         //testClassAnotation(user);
    14         //testFieldAnotation(user);
    15         testMethodAnnotation(user);
    16     }
    17     
    18     
    19     /**
    20      * 测试注释类的注解
    21      * @param user
    22      */
    23     public static void testClassAnotation(User user){
    24         //获取对象的类
    25         Class cls=user.getClass();
    26         //判断当前的类的对象是否是这个注解注释的
    27         boolean flag =cls.isAnnotationPresent(MyAnnotationByClass.class);
    28         if(flag){
    29             //如果是这个注解注释的,则获取这个注解对象
    30             MyAnnotationByClass myAnnotationByClass=(MyAnnotationByClass) cls.getAnnotation(MyAnnotationByClass.class);
    31             //获取注解的属性值
    32             String name=myAnnotationByClass.name();
    33             SEX sex=myAnnotationByClass.xingbie();
    34             int age=myAnnotationByClass.age();
    35             System.out.println("AnotationTest.testClassAnotation()"+name);
    36             System.out.println("AnotationTest.testClassAnotation()"+sex);
    37             System.out.println("AnotationTest.testClassAnotation()"+age);
    38         }
    39         
    40     }
    41     
    42     /**
    43      *测试属性的注解
    44      * @param user
    45      */
    46     public static void testFieldAnotation(User user){
    47         //获取对象的类
    48         Class cls=user.getClass();
    49         //判断当前的类的对象是否是这个注解注释的
    50         Field[] fields=cls.getDeclaredFields();
    51         for(int i=0;i<fields.length;i++){
    52             Field fil=fields[i];
    53             boolean flag=fil.isAnnotationPresent(MyAnnotationByField.class);
    54             if(flag){
    55                 MyAnnotationByField myAnnotationByField=fil.getAnnotation(MyAnnotationByField.class);
    56                 //获取属性注解的值
    57                 String value=myAnnotationByField.value();
    58                 System.out.println("AnotationTest.testFieldAnotation()"+value);
    59             }
    60         }
    61     }
    62     
    63     
    64     /**
    65      * 测试方法注解
    66      * @param user
    67      */
    68     public static void testMethodAnnotation(User user){
    69 
    70         //获取对象的类
    71         Class cls=user.getClass();
    72         //获取所有的方法
    73         Method[] methods=cls.getDeclaredMethods();
    74         for(int i=0;i<methods.length;i++){
    75             Method method=methods[i];
    76             boolean flag=method.isAnnotationPresent(MyAnotationByMethod.class);
    77             if(flag){
    78                 //获取注解的值
    79                 MyAnotationByMethod myAnotationByMethod=method.getAnnotation(MyAnotationByMethod.class);
    80                 String fundcheck=myAnotationByMethod.fundcheck();
    81                 System.out.println("AnotationTest.testMethodAnnotation()"+fundcheck);
    82             }
    83             
    84         }
    85     }
    86 }
    View Code
  • 相关阅读:
    HDU 5828 Rikka with Sequence (线段树+剪枝优化)
    Educational Codeforces Round 5 E. Sum of Remainders (思维题)
    HDU 2256 Problem of Precision (矩阵快速幂)
    Codeforces 597C. Subsequences (树状数组+dp)
    Codeforces Round #292 (Div. 1) B. Drazil and Tiles (类似拓扑)
    HDU 5794 A Simple Chess (Lucas + dp)
    Codeforces Round #365 (Div. 2) D. Mishka and Interesting sum (离线树状数组+前缀xor)
    Codeforces Round #313 (Div. 2) E. Gerald and Giant Chess (Lucas + dp)
    进程内存空间的分布
    快排,堆排与归并排序
  • 原文地址:https://www.cnblogs.com/shangxiaofei/p/5692920.html
Copyright © 2011-2022 走看看