系统定义的注解
一个是@Override:只能用在方法之上的,用来告诉别人这一个方法是改写父类的。一个是@Deprecated:建议别人不要使用旧的API的时候用的,编译的时候会用产生警告信息,可以设定在程序里的所有的元素上.
一个是@SuppressWarnings:这一个类型可以来暂时把一些警告信息消息关闭.
如果不清楚上面三个类型的具体用法,各位可以baidu或google一下的,很简单的。
定义自己的注解
注解的强大之处是它不仅可以使java程序变成自描述的,而且允许程序员自定义注解。注解的定义和接口差不多,只是在interface前面多了一个“@”。
@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface Action {public String name() default "";public String value() default "";public String scope() default "prototype"; //singleton,prototype,request,session,四个作用于于
}
上面的代码是一个最简单的注解。这个注解没有属性。也可以理解为是一个标记注解。就象Serializable接口一样是一个标记接口,里面定义了几个属性。
使用下面的格式使用
@Action("testAction")
public class TestAction {
}
看了上面的代码,大家可能有一个疑问。怎么没有使用value,而直接就写”abc”了。那么”abc”到底传给谁了。其实这里有一个约定。如果没有写属性名的值,而这个注解又有value属性,就将这个值赋给value属性,如果没有,就出现编译错误。并且Action中的属性必须定义默认的值。
@Action(name="testService", scope = "prototype")
public class TestService {
}
对注解进行限制
使用普通注解对注解进行注解的方法和对类、接口进行注解的方法一样。所不同的是,J2SE5.0为注解单独提供了4种注解,即元注解。
它们是Target、Retention、Documented和Inherited。下面就分别介绍这4种注解。
@Target里面的ElementType是用来指定Annotation类型可以用在哪一些元素上的.说明一下:TYPE(类型), FIELD(属性), METHOD(方法), PARAMETER(参数), CONSTRUCTOR(构造函数),LOCAL_VARIABLE(局部变量), ANNOTATION_TYPE,PACKAGE(包),其中的TYPE(类型)是指可以用在Class,Interface,Enum和Annotation类型上.
- @Target(ElementType.TYPE)
- @Retention(RetentionPolicy.RUNTIME)
- @Documented
- public @interface Action {
- public String name() default "";
- public String value() default "";
- public String scope() default "prototype"; //singleton,prototype,request,session,四个作用于于
- }
另外,从1的源代码可以看出,@Target自己也用了自己来声明自己,只能用在ANNOTATION_TYPE之上.
如果一个Annotation类型没有指明@Target使用在哪些元素上,那么它可以使用在任何元素之上,这里的元素指的是上面的八种类型.
举几个正确的例子:
@Target(ElementType.METHOD)
@Target(value=ElementType.METHOD)
@Target(ElementType.METHOD,ElementType.CONSTRUCTOR)
具体参考一下javadoc文档
@Retention里面的值
SOURCE代表的是这个Annotation类型的信息只会保留在程序源码里,源码如果经过了编译之后,Annotation的数据就会消失,并不会保留在编译好的.class文件里面。
ClASS的意思是这个Annotation类型的信息保留在程序源码里,同时也会保留在编译好的.class文件里面,在执行的时候,并不会把这一些信息加载到虚拟机(JVM)中去.注意一下,当你没有设定一个Annotation类型的Retention值时,系统默认值是CLASS.
第三个,是RUNTIME,表示在源码、编译好的.class文件中保留信息,在执行的时候会把这一些信息加载到JVM中去的. 要用java的反射机制读取注解,必须定义成这个类型
另外一点,如果需要把Annotation的数据继承给子类,那么就会用到@Inherited这一个Annotation类型.
注解的使用实例
定义一个注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Action {
public String name() default "";
public String value() default "";
public String scope() default "prototype"; //singleton,prototype,request,session,四个作用于于
}
@Action("testAction")
public class TestAction {
}
使用注解
public static AnnotationBean resolve(Class claz,Annotation anno){
AnnotationBean bean = null;
String classPath = claz.getName();
String name = null;
String value = null;
String scope = null;
String annoName = "";
if(Action.class.equals(anno.annotationType())){
Action action = (Action)anno;
annoName = "Action";
name = action.name();
value = action.value();
scope = action.scope();
}else if(Service.class.equals(anno.annotationType())){
Service action = (Service)anno;
annoName = "Service";
name = action.name();
value = action.value();
scope = action.scope();
}else if(Dao.class.equals(anno.annotationType())){
Dao action = (Dao)anno;
annoName = "Dao";
name = action.name();
value = action.value();
scope = action.scope();
}
if("".equals(name)){
name = value;
}
if(null!=name&&!"".equals(name)){
bean = new AnnotationBean(annoName,name,classPath,scope);
}
// System.out.println(name+"####"+scope+"$$$$$$$"+classPath);
return bean;
}
public static Map<String,AnnotationBean> resolve(Class claz){
Map<String,AnnotationBean> map = new HashMap<String,AnnotationBean>();
Annotation[] annotations = claz.getAnnotations();
AnnotationBean bean = null;
for(int i = 0 ; i<annotations.length ; i++){
Annotation anno = annotations[i];
bean = resolve(claz,anno);
if(bean!=null){
map.put(bean.getName(), bean);
}
}
return map;
}
解析这个类的注解
注解的本质:注解是一种类型
JDK5.0中的类型:1、类(class)2、接口(interface)3、枚举(enum)4、注解(Annotation)
因此,注解与其他3种类型一样,都可以定义、使用,以及包含有自己的属性、方法