最后有一点需要注意的是:
当我们使用@interface关键字定义一个注解时,该注解隐含地继承了java.lang.annotation.Annotation接口;如果我们定义了一个接口,并且让该接口继承自Annotation,那么我们所定义的接口依然还是接口而不是注解;Annotation本身是接口而不是注解。可以与Enum类比。
第2集
本集主要讲述@Retention及@Target2个注解,顺带提一下@Documented这个注解
1.关于@Retention这个注解
Retention翻译成中文是“保留”的意思,RetentionPolicy是“保留策略”。
简要描述:指示注解类型的注解要保留多久。如果注解类型声明中不存在 Retention 注解,则保留策略默认为 RetentionPolicy.CLASS。
每一个Retention都要给他一个RetentionType,RetentionType是一个枚举类型(具体可以查看API文档),它有3种取值:SOURCE,CLASS,RUNTIME,区别如下:
(a)SOURCE:表示该注解只会存在于JAVA源文件中,不会编译到class文件里面去,更不会在运行期通过反射的方式获 取到。
(b)CLASS:表示该注解会随着JAVA源代码一起编译到class文件里面去,但不会在运行期通过反射的方式获取到。
(c)RUNTIME:表示该注解会随着JAVA源代码一起编译到class文件里面去,并且会在运行期通过反射的方式获取到。
请看一个示例:
首先定义一个注解:
- package com.shengsiyuan.annotation;
-
- import java.lang.annotation.Retention;
- import java.lang.annotation.RetentionPolicy;
-
-
-
-
-
- @Retention(RetentionPolicy.RUNTIME)
- public @interface MyAnnotation
- {
- String hello() default "shengsiyuan";
-
- String world();
- }
package com.shengsiyuan.annotation;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
//注解也可以修饰注解,该注解修饰下面自定义的注解,通过设定
//RetentionPolicy的值为RUNTIME表示该自定义的注解会被编
//译到CLASS文件当中,而且可以在运行期通过反射的方式获取到(具体请查看一遍API文档,很有必要!)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation
{
String hello() default "shengsiyuan";
String world();
}
然后定义一个类,用这个Annotation去修饰
- package com.shengsiyuan.annotation;
-
- @MyAnnotation(hello = "beijing", world = "shanghai")
- public class MyTest
- {
-
- @MyAnnotation(hello = "tianjin", world = "shangdi")
- @Deprecated
- @SuppressWarnings("unchecked")
- public void output()
- {
- System.out.println("output something!");
- }
- }
package com.shengsiyuan.annotation;
@MyAnnotation(hello = "beijing", world = "shanghai")
public class MyTest
{
//一个方法可以被多个注解所修饰。
@MyAnnotation(hello = "tianjin", world = "shangdi")
@Deprecated
@SuppressWarnings("unchecked")
public void output()
{
System.out.println("output something!");
}
}
接着定义一个类,并通过反射相关API去获得自定义注解的相关信息
- package com.shengsiyuan.annotation;
-
- import java.lang.annotation.Annotation;
- import java.lang.reflect.Method;
-
-
- public class MyReflection
- {
- public static void main(String[] args) throws Exception
- {
- MyTest myTest = new MyTest();
-
- Class<MyTest> c = MyTest.class;
-
- Method method = c.getMethod("output", new Class[]{});
-
-
- if(method.isAnnotationPresent(MyAnnotation.class))
- {
- method.invoke(myTest, new Object[]{});
-
- MyAnnotation myAnnotation = method.getAnnotation(MyAnnotation.class);
-
- String hello = myAnnotation.hello();
- String world = myAnnotation.world();
-
- System.out.println(hello + ", " + world);
- }
-
-
-
- Annotation[] annotations = method.getAnnotations();
-
- for(Annotation annotation : annotations)
- {
- System.out.println(annotation.annotationType().getName());
- }
- }
- }
package com.shengsiyuan.annotation;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
//该类拿到修饰MyTest里方法的Annotation
public class MyReflection
{
public static void main(String[] args) throws Exception
{
MyTest myTest = new MyTest();
Class<MyTest> c = MyTest.class;
Method method = c.getMethod("output", new Class[]{});
//能够进入到if语句里面来说明MyAnnotation的RetentionPolicy的值为Runtime(为什么请查API文档!)
if(method.isAnnotationPresent(MyAnnotation.class))
{
method.invoke(myTest, new Object[]{});
MyAnnotation myAnnotation = method.getAnnotation(MyAnnotation.class);
String hello = myAnnotation.hello();
String world = myAnnotation.world();
System.out.println(hello + ", " + world);
}
//只会得到Myannotation和Deprecated两个Annotation,因为只有这两个Annotation的RetentionPolicy
//的值为Runtime,只有RetentionPolicy的值为Runtime才会在运行期通过反射相关API拿到Annotation的相关信息。
Annotation[] annotations = method.getAnnotations();
for(Annotation annotation : annotations)
{
System.out.println(annotation.annotationType().getName());
}
}
}
2.关于@Target这个注解(建议去读一读API文档,介绍的很详细)
简要描述:指示注解类型所适用的程序元素的种类。如果注解类型声明中不存在 Target 元注解,则声明的类型可以用在任一程序元素上。
每一个Target都要给他一个ElementType,ElementType是一个枚举类型(具体可以查看API文档),它有8种取值:SOURCE,CLASS,RUNTIME,区别如下:
(a)ANNOTATION_TYPE:表示该注解可以去修饰另外一个注解
(b)COUNSTRUCTOR:表示该注解可以修饰构造方法
(c)FIELD:表示该注解可以修饰成员变量
(d)LOCAL_VARIABLE:表示该注解可以修饰局部变量
(e)METHOD:表示该注解可以修饰普通方法
(f)PACKAGE:表示该注解可以修饰包
(g)PARAMETER:表示该注解可以修饰方法参数
(h)TYPE:表示该注解可以修饰类、接口(包括注解类型)或枚举声明
请看示例:
- package com.shengsiyuan.annotation;
-
- import java.lang.annotation.ElementType;
- import java.lang.annotation.Target;
-
- @Target(ElementType.METHOD)
- public @interface MyTarget
- {
- String value();
- }
package com.shengsiyuan.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Target;
@Target(ElementType.METHOD)//表示该自定义注解只能用于修饰方法
public @interface MyTarget
{
String value();
}
接着定义一个类:
- package com.shengsiyuan.annotation;
-
- public class MyTargetTest
- {
- @MyTarget("hello")
- public void doSomething()
- {
- System.out.println("hello world");
- }
- }
package com.shengsiyuan.annotation;
public class MyTargetTest
{
@MyTarget("hello")
public void doSomething()
{
System.out.println("hello world");
}
}
当把该自定义的注解放到方法上面后编译器不报错时,说明我们的实验是成功的(不需要写main方法进行测试)
对以上2个注解的总结:Retention与Target都是注解,Retention与RetentionPolicy搭配,Target与ElementType搭配。
3.关于@Documented(了解就行)
不多做描述,请看示例:
- package com.shengsiyuan.annotation;
-
- import java.lang.annotation.Documented;
-
- @Documented
- public @interface DocumentedAnnotation
- {
- String hello();
- }
package com.shengsiyuan.annotation;
import java.lang.annotation.Documented;
@Documented //当一个注解被@Documented 修饰后表示被该注解修饰的对象(类或方法或其他)在生成JAVA DOC文档时,该注解会被加到修饰的对象的上面
public @interface DocumentedAnnotation
{
String hello();
}
然后用该注解去修饰某个方法
- package com.shengsiyuan.annotation;
-
- public class DocumentedTest
- {
- @DocumentedAnnotation(hello = "welcome")
- public void method()
- {
- System.out.println("hello world");
- }
- }
package com.shengsiyuan.annotation;
public class DocumentedTest
{
@DocumentedAnnotation(hello = "welcome")
public void method()
{
System.out.println("hello world");
}
}
当对DocumentedTest所在的包或工程生成JAVA DOC文档的时候,会发现自定义的注解会出现在method方法的上面