2. 四种元注解

元注解,说白了,就是JDK自带的注解,这些注解是干嘛的呢?其实就是在我们自定义注解时,注解到我们自定义的注解上的,举个例子:

@Target(ElementType.TYPE) public @interface Table { public String tableName() default "className"; }

大家先不要管语法,Table其实就是我自定义的一个注解,可以@Table这样使用了,那么这个@Target其实就是元注解了。JDK自带的元注解如下:

@Target@Retention@Documented@Inherited

@Target

用于描述注解的使用范围,有一个枚举ElementType来指定,具体如下:

CONSTRUCTOR:用于描述构造器FIELD:用于描述域LOCAL_VARIABLE:用于描述局部变量METHOD:用于描述方法PACKAGE:用于描述包PARAMETER:用于描述参数TYPE:用于描述类、接口(包括注解类型) 或enum声明

eg:

@Target(ElementType.METHOD) public @interface Dog { }

大家可以看到,这个Dog注解,只能用到方法上面 。

1). @Retention

表示需要在什么级别保存该注释信息,用于描述注解的生命周期,也是一个枚举RetentionPoicy来决定的,这个枚举我不列出来了,包括这个注解的具体怎么决定注解的生命周期我也不多讲,因为根据小弟这么多年使用的经验,都是填的RetentionPoicy.RUNTIME,填这个值注解处理器才能通过反色拿到注解信息,实现自己的语义,所以大家都填RetentionPoicy.RUNTIME就可以了,想扩展了解的自行google..

2). @Documented

如果用javadoc生成文档时,想把注解也生成文档,就带这个。(话说老铁们都是怎么写文档的,小弟表示从不写文档... ^_^)

3). @Inherited

@Inherited 元注解是一个标记注解,@Inherited阐述了某个被标注的类型是被继承的。如果一个使用了@Inherited修饰的annotation类型被用于一个class,则这个annotation将被用于该class的子类。注意,@Inherited annotation类型是被标注过的class的子类所继承。类并不从它所实现的接口继承annotation,方法并不从它所重载的方法继承annotation。

其实这几个注解只有一个有点用,就是@Target,大家稍微注意下就行了。注解不是你想加哪就加哪的。

java 自定义注解在方法执行完后获取方法中的变量并执行其它操作 java自定义注解中的元注解_java自定义元注解

3. 自定义注解

上面把元注解说了,来讲下如何自定义注解,顺带提一下,自定义的注解又叫组合注解.然后说下自定义注解格式:

java 自定义注解在方法执行完后获取方法中的变量并执行其它操作 java自定义注解中的元注解_元注解_02

1)固定格式,不要搞事.

2)只能用public或默认(default)这两个访问权修饰.例如,String value();这里把方法设为defaul默认类型;

3)定义注解参数的类型,只能为:

所有基本数据类型(int,float,boolean,byte,double,char,long,short)String类型Class类型enum类型Annotation类型以上所有类型的数组

方法名,就是这个注解的支持的属性名,像常见的value.表示该属性名在不指定时的默认值,可以不要。

好了,举个例子:

@Target(ElementType.FIELD) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface Dog { public String dog() default ""; }

好了,大家可以看到,这样就定义了一个注解了。

java 自定义注解在方法执行完后获取方法中的变量并执行其它操作 java自定义注解中的元注解_自定义注解_03

自定义注解有两种用法:

自定义注解处理器专门为你自定义的注解实现语义(比如加了@Dog注解的方法,在调用的时候,全部都打印出调用信息到日志系统) ,后一篇博客讲自定义注解处理器。配合Spring使用.

在讲如何配合Spring使用前,先说一个知识点,大家可能会忽略掉,那就是注解是可以组合的,如上图,在自定义注解时,不仅可以加元注解,还可以加其他自定义注解 。这就好玩了,我举个例子:

@Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Documented @Component public @interface Dog { public String dog() default ""; }

我为刚刚的@Dog注解加了@Component,那么这个注解就会有@Component的功能了(被Spring当做bean加入到Spring容器里).现在在顺带说下属性修饰符public和默认的区别

public修饰的属性能被覆盖,默认的则不能. 举个例子:

@Target(ElementType.FIELD) @Retention(RetentionPolicy.RUNTIME) @Documented @Component @Dog public @interface StupidDog { public String dog() default "我是一条StupidDog"; }

我现在又定义了一个StupidDog注解,这个注解覆盖掉了@Dog的dog属性,把默认值从""改成了"我是一条StupidDog"..

好了,回到话题上,知道了组合注解可以拥有被注解的注解的功能,还拿刚才那个加了@Component注解的@Dog注解来说,当某个类被加了这个注解,比如:

@Dog public class TestService { ... }

我们知道这个TestService在Spring的环境下会被加到Spring容器里成为Bean,这个作用可大了!这可以把Spring容器里的bean区分开来,比如如下代码片段:

// ctx 为Spring的ApplicationContext // // 获取所有带有 Dog 注解的 Spring Bean Map serviceBeanMap = ctx.getBeansWithAnnotation(Dog.class);