本文转发自Jason’s Blog,原文连接 http://www.jasongj.com/2016/01/17/Java1_注解Annotationjava
Annotation是Java5开始引入的特性。它提供了一种安全的相似于注释和Java doc的机制。实事上,Annotation已经被普遍用于各类Java框架,如Spring,Jersey,JUnit,TestNG。注解至关因而一种嵌入在程序中的元数据,可使用注解解析工具或编译器对其进行解析,也能够指定注解在编译期或运行期有效。这些元数据与程序业务逻辑无关,而且是供指定的工具或框架使用的。数组
元注解的做用就是负责注解其余注解。Java5定义了4个标准的Meta Annotation类型,它们被用来提供对其它 Annotation类型做说明。安全
@Target
说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在Annotation类型的声明中使用了@Target
可更加明晰其修饰的目标。bash
@Target
做用:用于描述注解的使用范围,即被描述的注解能够用在什么地方框架
@Target
取值(ElementType)ide
CONSTRUCTOR
:用于描述构造器FIELD
:用于描述域LOCAL_VARIABLE
:用于描述局部变量METHOD
:用于描述方法PACKAGE
:用于描述包PARAMETER
:用于描述参数TYPE
:用于描述类、接口(包括注解类型) 或enum声明@Retention
定义了该Annotation的生命周期:某些Annotation仅出如今源代码中,而被编译器丢弃;而另外一些却被编译在class文件中;编译在class文件中的Annotation可能会被虚拟机忽略,而另外一些在class被装载时将被读取(请注意并不影响class的执行,由于Annotation与class在使用上是被分离的)。@Retention
有惟一的value做为成员。工具
@Retention
做用:表示须要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)spa
@Retention
取值来自java.lang.annotation.RetentionPolicy
的枚举类型值.net
@Documented
用于描述其它类型的annotation应该被做为被标注的程序成员的公共API,所以能够被例如javadoc此类的工具文档化。@Documented
是一个标记注解,没有成员。code
@Inherited
是一个标记注解。若是一个使用了@Inherited
修饰的annotation类型被用于一个class,则这个Annotation将被用于该class的子类。
在实际项目中,常常会碰到下面这种场景,一个接口的实现类或者抽象类的子类不少,常常须要根据不一样状况(好比根据配置文件)实例化并使用不一样的子类。典型的例子是结合工厂使用职责链模式。
此时,能够为每一个实现类加上特定的Annotation,并在Annotation中给该类取一个标识符,应用程序可经过该标识符来判断应该实例化哪一个子类。
下面这个例子,定义了一个名为Component的Annotation,它包含一个名为identifier的成员变量。
package com.jasongj.annotation; import java.lang.annotation.Documented; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; import java.lang.annotation.ElementType; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Documented @Inherited public @interface Component { String identifier () default ""; }
对于上文所说的实现类加上@Component
package com.jasongj; import com.jasongj.annotation.Component; @Component(identifier="upper") public class UpperCaseComponent { public String doWork(String input) { if(input != null) { return input.toUpperCase(); } else { return null; } } }
应用程序中能够经过反射获取UpperCaseComponent对应的identifier
package com.jasongj; import com.jasongj.annotation.Component; public class Client { public static void main(String[] args) { try { Class componentClass = Class.forName("com.jasongj.UpperCaseComponent"); if(componentClass.isAnnotationPresent(Component.class)) { Component component = (Component)componentClass.getAnnotation(Component.class); String identifier = component.identifier(); System.out.println(String.format("Identifier for " + "com.jasongj.UpperCaseComponent is ' %s '", identifier)); } else { System.out.println("com.jasongj.UpperCaseComponent is not annotated by" + " com.jasongj.annotation.Component"); } } catch (ClassNotFoundException ex) { ex.printStackTrace(); } } }
结果以下
Identifier for com.jasongj.UpperCaseComponent is ' upper '
若是把@Component
的@Retention
设置为 RetentionPolicy.SOURCE
或者RetentionPolicy.CLASS
,则会获得以下结果,验证了上文中对@Retention
的描述
com.jasongj.UpperCaseComponent is not annotated by com.jasongj.annotation.Component
Annotation的语法比较简单,除了@符号的使用外,他基本与Java固有的语法一致,JavaSE中内置三个标准Annotation,定义在java.lang
中:
@Override
是一个标记型Annotation,说明了被标注的方法覆盖了父类的方法,起到了断言的做用。若是给一个非覆盖父类方法的方法添加该Annotation,编译器将报编译错误。它有两个典型的使用场景,一是在试图覆盖父类方法却写错了方法名时报错,二是删除已被子类覆盖(且用Annotation修饰)的父类方法时报错。@Deprecated
标记型Annotation,说明被修改的元素已被废弃并不推荐使用,编译器会在该元素上加一条横线以做提示。该修饰具备必定的“传递性”:若是咱们经过继承的方式使用了这个弃用的元素,即便继承后的元素(类,成员或者方法)并未被标记为@Deprecated
,编译器仍然会给出提示。@SuppressWarnnings
用于通知Java编译器关闭对特定类、方法、成员变量、变量初始化的警告。此种警告通常表明了可能的程序错误,例如当咱们使用一个generic collection类而未提供它的类型时,编译器将提示“unchecked warning”的警告。一般当这种状况发生时,咱们须要查找引发警告的代码,若是它真的表示错误,咱们就须要纠正它。然而,有时咱们没法避免这种警告,例如,咱们使用必须和非generic的旧代码交互的generic collection类时,咱们没法避免这个unchecked warning,此时能够在调用的方法前增长@SuppressWarnnings
通知编译器关闭对此方法的警告。@SuppressWarnnings
不是标记型Annotation,它有一个类型为String[]的成员,这个成员的值为被禁止的警告名。常见的警告名为下。
unchecked
执行了未检查的转换时的警告。例如当使用集合时没有用泛型来指定集合的类型finally
finally子句不能正常完成时的警告fallthrough
当switch程序块直接通往下一种状况而没有break时的警告deprecation
使用了弃用的类或者方法时的警告seriel
在可序列化的类上缺乏serialVersionUID时的警告path
在类路径、源文件路径等中有不存在的路径时的警告all
对以上全部状况的警告@interface
而非interface
。注意开头的@
符号