`

Java注解之Retention、Documented、Inherited介绍

 
阅读更多

Retention注解

Retention(保留)注解说明,这种类型的注解会被保留到那个阶段. 有三个值:
1.RetentionPolicy.SOURCE —— 这种类型的Annotations只在源代码级别保留,编译时就会被忽略
2.RetentionPolicy.CLASS —— 这种类型的Annotations编译时被保留,在class文件中存在,但JVM将会忽略
3.RetentionPolicy.RUNTIME —— 这种类型的Annotations将被JVM保留,所以他们能在运行时被JVM或其他使用反射机制的代码所读取和使用.
示例演示了 RetentionPolicy.RUNTIME 的声明:

@Retention(RetentionPolicy.RUNTIME)

public @interface Test_Retention {

   String doTestRetention();

}

 在这个示例中, @Retention(RetentionPolicy.RUNTIME)注解表明 Test_Retention注解将会由虚拟机保留,以便它可以在运行时通过反射读取.

 

Documented 注解

Documented 注解表明这个注解应该被 javadoc工具记录. 默认情况下,javadoc是不包括注解的. 但如果声明注解时指定了 @Documented,则它会被 javadoc 之类的工具处理, 所以注解类型信息也会被包括在生成的文档中.

示例进一步演示了使用 @Documented:

@Documented

public @interface Test_Documented {

   String doTestDocument();

}

 

public class TestAnnotations {

   public static void main(String arg[]) {

      new TestAnnotations().doSomeTestRetention();

      new TestAnnotations().doSomeTestDocumented();

   }

   @Test_Retention (doTestRetention="保留注解信息测试")

   public void doSomeTestRetention() {

      System.out.printf("测试注解类型 'Retention'");

   }

   @Test_Documented(doTestDocument="Hello document")

   public void doSomeTestDocumented() {

      System.out.printf("测试注解类型 'Documented'");

   }

}

 现在,如果你使用 javadoc命令生成 TestAnnotations.html文件,你将看到类似于图的结果.



 

从截图可以看到,文档中没有 doSomeTestRetention() 方法的 annotation-type信息()方法. 但是, doSomeTestDocumented() 方法的文档提供了注解的描述信息. 这是因为 @Documented标签被加到了Test_Documented注解上. 之前的注解Test_Retention并没有指定 @Documented 标记(tag).

Inherited 注解

这是一个稍微复杂的注解类型. 它指明被注解的类会自动继承. 更具体地说,如果定义注解时使用了 @Inherited 标记,然后用定义的注解来标注另一个父类, 父类又有一个子类(subclass),则父类的所有属性将被继承到它的子类中. 在示例7中,你会看到使用 @Inherited 标签的好处.

@Inherited

public @interface MyParentObject { 

      boolean isInherited() default true;

      String doSomething() default "Do what?";

}

 

@MyParentObject

public Class MyChildObject {

}

 正如你看到的,你不需要在实现类中定义接口方法. 因为使用 @Inherited标记,这些都自动继承了. 如果你使用一种古老的方式定义实现类,会是什么样子呢? 看看下面这张 古老的实现方式吧:

public class MyChildObject implements MyParentObject {

   public boolean isInherited() {

      return false;

   }

   public String doSomething() {

      return "";

   }

   public boolean equals(Object obj) {

      return false;

   }

   public int hashCode() {

      return 0;

   }

   public String toString() {

      return "";

   }

   public Class annotationType() {

      return null;

   }

}

 看到的区别吗? 可以看到,你必须实现父接口的所有方法. 除了isInherited()和从myParentObject doSomething()方法外,你还需要实现 java.lang.Object的 equals(),toString()和hasCode()方法. 还有 java.lang.annotation.Annotation 类的 annotationType()方法. 不管你是不是想要实现这些方法,你必须在继承的对象中包含这些.

 

from:http://www.jb51.net/article/55371.htm

   我们自定义注解(Annotation)时,把自定义的注解标注在父类上不会被子类所继承,但是我们可以在定义注解时给我们自定义的注解标注一个@Inherited注解来实现注解继承。 

自定义的注解代码如下: 
Java代码  收藏代码
  1. package com.xdf.annotation;  
  2.   
  3. import java.lang.annotation.Inherited;  
  4. import java.lang.annotation.Retention;  
  5.   
  6. @Inherited  
  7. @Retention(java.lang.annotation.RetentionPolicy.RUNTIME)  
  8.   
  9. public @interface InheritedAnnotation {  
  10.   
  11.     String value();  
  12.   
  13. }  


接着定义一个抽象父类 
代码如下: 
Java代码  收藏代码
  1. package com.xdf.annotation;  
  2.   
  3. public abstract class AbstractParent {  
  4.   
  5.     @InheritedAnnotation(value = "parent abstractMethod ")  
  6.      public abstract void abstractMethod();  
  7.       
  8.     @InheritedAnnotation(value = "Parent's doExtends")  
  9.      public void doExtends() {  
  10.       System.out.println(" AbstractParent doExtends ...");  
  11.      }  
  12.   
  13.       
  14. }  


我们把自定义的注解父类里的方法上。 

接着定义一个继承抽象父类(AbstractParent)的子类 
代码如下: 
Java代码  收藏代码
  1. package com.xdf.annotation;  
  2.   
  3. public class SubClassImpl extends AbstractParent{  
  4.   
  5.     @Override  
  6.     public void abstractMethod() {  
  7.     System.out.println("子类实现抽象父类的抽象方法");  
  8.           
  9.     }  
  10.   
  11. }  


在子类中实现了abstractMethod抽象方法,没有重写doExtends方法。 

测试类代码如下: 
Java代码  收藏代码
  1. package com.xdf.annotation;  
  2.   
  3. import java.lang.reflect.Method;  
  4.   
  5. public class InheritedAnnotationTest {  
  6.   
  7.     public static void main(String[] args) throws SecurityException, NoSuchMethodException {  
  8.           
  9.           Class<SubClassImpl> clazz=SubClassImpl.class;  
  10.            
  11.             
  12.             //abstractMethod  
  13.           Method method = clazz.getMethod("abstractMethod"new Class[]{});  
  14.           if(method.isAnnotationPresent(InheritedAnnotation.class)){  
  15.               InheritedAnnotation ma = method.getAnnotation(InheritedAnnotation.class);  
  16.            System.out.println("子类实现的抽象方法继承到父类抽象方法中的Annotation,其信息如下:");  
  17.            System.out.println(ma.value());  
  18.           }else{  
  19.            System.out.println("子类实现的抽象方法没有继承到父类抽象方法中的Annotation");  
  20.           }  
  21.             
  22.             
  23.             
  24.           
  25.           Method methodOverride = clazz.getMethod("doExtends"new Class[]{});  
  26.           if(methodOverride.isAnnotationPresent(InheritedAnnotation.class)){  
  27.               InheritedAnnotation ma = methodOverride.getAnnotation(InheritedAnnotation.class);  
  28.            System.out.println("子类doExtends方法继承到父类doExtends方法中的Annotation,其信息如下:");  
  29.            System.out.println(ma.value());  
  30.           }else{  
  31.            System.out.println("子类doExtends方法没有继承到父类doExtends方法中的Annotation");  
  32.           }  
  33.   
  34.           
  35.   
  36.     }       
  37.                
  38. }  

运行结果如下: 
Java代码  收藏代码
  1. 子类实现的抽象方法没有继承到父类抽象方法中的Annotation  
  2. 子类doExtends方法继承到父类doExtends方法中的Annotation,其信息如下:  
  3. Parent's doExtends  


从以上代码运行的结果可以得到如下结论: 
1.如果子类继承父类,并且重写了父类中的带有注解的方法,那么父类方法上的注解是不会被子类继承的。 
2.如果子类继承父类,但是没有重写父类中带有注解的方法,那么父类方法上的注解会被子类继承,就是说在子类中可以得到父类方法上的注解。 

但是.....但是....当我把自定义的注解上的@Inherited注解去掉运行,结果还是一样,也就是说这个@Inherited注解根本没有起作用。这是什么神马情况呢? 

接着我把没有标注@Inherited注解的自定义的注解标注在类级别上(不是方法级别上),把抽象父类改成下面这样: 
Java代码  收藏代码
  1. package com.xdf.annotation;  
  2.   
  3.   
  4. @InheritedAnnotation(value="parent")   //把自定义注解标注在父类上  
  5. public abstract class AbstractParent {  
  6.   
  7.     @InheritedAnnotation(value = "parent abstractMethod ")  
  8.      public abstract void abstractMethod();  
  9.       
  10.     @InheritedAnnotation(value = "Parent's doExtends")  
  11.      public void doExtends() {  
  12.       System.out.println(" AbstractParent doExtends ...");  
  13.      }  
  14.   
  15.       
  16. }  

然后在测试类的main方法里加了如下测试代码: 
Java代码  收藏代码
  1. if(clazz.isAnnotationPresent(InheritedAnnotation.class)){  
  2.               InheritedAnnotation cla = clazz.getAnnotation(InheritedAnnotation.class);  
  3.            System.out.println("子类继承到父类类上Annotation,其信息如下:");  
  4.            System.out.println(cla.value());  
  5.           }else{  
  6.               System.out.println("子类没有继承到父类类上Annotation");  
  7.           }  



这是出现情况了,运行main方法得到结果: 
Java代码  收藏代码
  1. 子类实现的抽象方法没有继承到父类抽象方法中的Annotation  
  2. 子类doExtends方法继承到父类doExtends方法中的Annotation,其信息如下:  
  3. Parent's doExtends  
  4. 子类没有继承到父类类上Annotation  


从运行结果中可以发现子类并没有继承父类类级别的注解,于是我又把@Inherited注解标注在自定义注解上,然后运行一下,得到如下结果: 

Java代码  收藏代码
  1. 子类实现的抽象方法没有继承到父类抽象方法中的Annotation  
  2. 子类doExtends方法继承到父类doExtends方法中的Annotation,其信息如下:  
  3. Parent's doExtends  
  4. 子类继承到父类类上Annotation,其信息如下:  
  5. parent  



注意运行结果,子类继承了父类类级别的注解了。 

这说明什么呢? 
说明这种标有@Inherited注解的自定义的注解运用到类级别上和方法级别上是不一样的,如果把标有@Inherited注解的自宝义的注解标注在类级别上,子类则可以继承父类类级别的注解,反之,则不行。

 from:http://xiangdefei.iteye.com/blog/1044199

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  • 大小: 43.5 KB
分享到:
评论

相关推荐

    Java元注解是Java注解的重要特性,下面我就来详解常见的4大Java元注解

    Java 元注解包括四大类:@Retention、@Target、@Documented、@Inherited 等。 1.@Retention @Retention 用于指定注解的保留期限,可以设置为 SOURCE、CLASS 或 RUNTIME。 * RetentionPolicy.SOURCE:注解只保留在...

    JAVA 注解示例 详解

    4. **元注解**:元注解是用于注解其他注解的注解,如`@Retention`、`@Target`、`@Documented`和`@Inherited`: - `@Retention`:指定注解的生命周期,可以是`SOURCE`(源码级)、`CLASS`(类文件级)或`RUNTIME`...

    入门级java 注解学习示例代码

    Java注解(Annotation)是Java语言的一个重要特性,它为元数据提供了强大的支持。元数据是一种描述数据的数据,它提供了一种安全的方式,使我们可以在不改变程序代码本身的情况下,向编译器或运行时环境提供额外的...

    Java自定义注解实例

    3. **元注解**: 元注解是用于注解其他注解的注解,如`@Retention`、`@Target`、`@Documented`和`@Inherited`。它们控制注解的生命周期和使用范围。 ### 二、自定义注解 1. **创建**: 自定义注解通过定义一个接口...

    java 注解开发(包含详细注释,项目可以直接运行)

    - **元注解**:用于定义注解的注解,如`@Retention`、`@Target`、`@Documented`、`@Inherited`。 2. **注解的保留策略** - `@Retention(RetentionPolicy.SOURCE)`:注解只存在于源码阶段,编译后不保留。 - `@...

    java 注解

    Java注解是Java编程语言中的一个重要特性,它允许程序员在代码中嵌入元数据(metadata),这些元数据不直接影响程序的执行,但可以被编译器或运行时环境用来进行各种处理,如验证、动态代理、持久化、资源管理等。...

    对spring做java注解扩展

    例如,`@Target`、`@Retention`、`@Documented`和`@Inherited`是Java提供的元注解,它们分别定义了注解的应用目标、保留策略、是否包含在Javadoc中以及是否可以被子类继承。Spring还引入了`@Repeatable`,允许同一...

    Spring java注解,元注解和自定义注解

    自定义注解是Java和Spring中非常强大的功能之一。通过定义自定义注解,开发者可以为程序添加更多特定意义的信息,并通过Spring的AOP(面向切面编程)功能实现特定的行为。 1. **定义自定义注解** - 使用@interface...

    深入浅出Java注解(注解概述+注解的本质+反射注解信息+元注解+属性的数据类型及特别的属性:value和数组)

    "Java 注解详解" Java 注解是 Java 中最重要却最容易被人遗忘的知识点。很多人不明白它是怎么起作用的,甚至有人把它和注释混淆。...通过本文的介绍,我们可以更好地理解 Java 注解的概念和使用方法。

    02-Java注解面试题(2题)-新增.pdf

    "Java注解面试题详解" 在 Java 中,注解(Annotation)是一种提供元数据的途径,用于关联信息和元数据。Java 5.0 中定义了 4 种标准的元注解(meta-annotation),它们被用来提供对其他注解的说明。 1. @Target @...

    Java注解学习

    ### Java注解学习 #### 知识点概览 Java注解是Java 5.0引入的一个重要特性,用于向代码添加元数据,而不影响其实际功能。它们为开发过程中的自动化工具、编译器检查、运行时元数据读取等提供了强大的功能。本文将...

    关于java注解比较详细的例子

    常见的元注解还包括`@Documented`(将注解包含在Javadoc中)、`@Inherited`(子类继承父类的注解)和`@Repeatable`(允许同一位置重复应用注解)。 5. **处理注解** 注解处理器是Java编译器的一部分,它们在编译...

    Java Annotation注解技术

    5. **元注解**:元注解是用于定义注解本身的注解,如`@Target`、`@Retention`、`@Documented`和`@Inherited`。它们提供了对注解的进一步控制,如注解可以应用的位置、保留策略等。 6. **标准注解**:Java预定义的...

    java 注解annotation的使用以及反射如何获取注解

    Java预定义了四个元注解:@Retention、@Target、@Documented和@Inherited。 - @Retention:定义注解的生命周期,有SOURCE、CLASS和RUNTIME三个策略。SOURCE表示注解只在源代码中存在,编译后会被丢弃;CLASS表示...

    关于java注解的使用

    自定义注解是Java注解的强大之处。自定义注解本质上是一个接口,使用`@interface`关键字声明。例如,我们可以定义一个名为`NewAnnotation`的注解: ```java public @interface NewAnnotation { // 可以定义成员...

    java 元注解 用于定义其它的注解

    @Retention 注解是 Java 中的一种元注解,用于指定注解的保留级别。RetentionPolicy 是一个枚举类型,它定义了被 @Retention 修饰的注解所支持的保留级别。RetentionPolicy 中有三个值:SOURCE、CLASS 和 RUNTIME。...

    Java元注解.docx

    Java 5引入了四个基本的元注解:@Documented、@Target、@Retention和@Inherited,而Java 8又添加了@Repeatable和@Native。 1. **@Documented**: @Documented注解是一个标记注解,不包含任何成员变量。当一个...

    最全Java注解图文超详解(建议收藏)

    Java预定义了几个元注解,如`@Retention`、`@Target`、`@Documented`和`@Inherited`,用于控制注解的行为。 9. **注解和反射**: 反射API可以用于在运行时检查类、接口、字段和方法的注解。`java.lang.reflect`...

    java注解详解

    包括 @Retention、@Target、@Document、@Inherited 四种。 1.1、@Retention:定义注解的保留策略 @Retention(RetentionPolicy.SOURCE) // 注解仅存在于源码中,在 class 字节码文件中不包含 @Retention...

Global site tag (gtag.js) - Google Analytics