Java元注解作用及使用
本节示例会用到自定义注解,不了解可先阅读学习《Java自定义注解》一节。
@Documented
@Documented 是一个标记注解,没有成员变量。用 @Documented 注解修饰的注解类会被 JavaDoc 工具提取成文档。默认情况下,JavaDoc 是不包括注解的,但如果声明注解时指定了 @Documented,就会被 JavaDoc 之类的工具处理,所以注解类型信息就会被包括在生成的帮助文档中。下面通过示例来了解它的用法,代码如下所示。
例 1
@Documented
@Target({ ElementType.TYPE, ElementType.METHOD })
public @interface MyDocumented {
public String value() default "这是@Documented注解";
}
测试类:
@MyDocumented
public class DocumentedTest {
/**
* 测试document
*/
@MyDocumented
public String Test() {
return "C语言中文网Java教程";
}
}
打开 Java 文件所在的目录,分别输入如下两条令行:
运行成功后,打开生成的帮助文档,可以看到在类和方法上都保留了 MyDocument 的注解信息。如下图所示:javac MyDocumented.java DocumentedTest.java
javadoc -d doc MyDocumented.java DocumentedTest.java


@Target
@Target 注解用来指定一个注解的使用范围,即被 @Target 修饰的注解可以用在什么地方。@Target 注解有一个成员变量(value)用来设置适用目标,value 是 java.lang.annotation.ElementType 枚举类型的数组,下表为 ElementType 常用的枚举常量。名称 | 说明 |
|---|---|
CONSTRUCTOR | 用于构造方法 |
FIELD | 用于成员变量(包括枚举常量) |
LOCAL_VARIABLE | 用于局部变量 |
METHOD | 用于方法 |
PACKAGE | 用于包 |
PARAMETER | 用于类型参数(JDK 1.8新增) |
TYPE | 用于类、接口(包括注解类型)或 enum 声明 |
例 2
自定义一个 MyTarget 注解,使用范围为方法,代码如下所示。
@Target({ ElementType.METHOD })
public @interface MyTarget {
}
class Test {
@MyTarget
String name;
}
如上代码第 6 行会编译错误,错误信息为:
The annotation @MyTarget is disallowed for this location
提示此位置不允许使用注解 @MyDocumented,@MyTarget 不能修饰成员变量,只能修饰方法。@Retention
@Retention 用于描述注解的生周期,也就是该注解被保留的时间长短。@Retention 注解中的成员变量(value)用来设置保留策略,value 是 java.lang.annotation.RetentionPolicy 枚举类型,RetentionPolicy 有 3 个枚举常量,如下所示。
SOURCE:在源文件中有效(即源文件保留)
CLASS:在 class 文件中有效(即 class 保留)
RUNTIME:在运行时有效(即运行时保留)
生周期大小排序为 SOURCE < CLASS < RUNTIME,前者能使用的地方后者一定也能使用。如果需要在运行时去动态获取注解信息,那只能用 RUNTIME 注解;如果要在编译时进行一些预处理操作,比如生成一些辅助代码(如 ButterKnife),就用 CLASS 注解;如果只是做一些检查性的操作,比如 @Override 和 @SuppressWarnings,则可选用 SOURCE 注解。
@Inherited
@Inherited 是一个标记注解,用来指定该注解可以被继承。使用 @Inherited 注解的 Class 类,表示这个注解可以被用于该 Class 类的子类。就是说如果某个类使用了被 @Inherited 修饰的注解,则其子类将自动具有该注解。
例 3
创建一个自定义注解,代码如下所示:
@Target({ ElementType.TYPE })
@Inherited
@Retention(RetentionPolicy.RUNTIME)
public @interface MyInherited {
}
测试类代码如下:
@MyInherited
public class TestA {
public static void main(String[] args) {
System.out.println(TestA.class.getAnnotation(MyInherited.class));
System.out.println(TestB.class.getAnnotation(MyInherited.class));
System.out.println(TestC.class.getAnnotation(MyInherited.class));
}
}
class TestB extends TestA {
}
class TestC extends TestB {
}
运行结果为:
@MyInherited()
@MyInherited()
@MyInherited()
@Repeatable
@Repeatable 注解是 Java 8 新增加的,它允许在相同的程序元素中重复注解,在需要对同一种注解多次使用时,往往需要借助 @Repeatable 注解。Java 8 版本以前,同一个程序元素前最多只能有一个相同类型的注解,如果需要在同一个元素前使用多个相同类型的注解,则必须使用注解“容器”。
例 4
Java 8 之前的做法:
public @interface Roles {
Role[] roles();
}
public @interface Roles {
Role[] value();
}
public class RoleTest {
@Roles(roles = {@Role(roleName = "role1"), @Role(roleName = "role2")})
public String doString(){
return "这是C语言中国网Java教程";
}
}
Java 8 之后增加了重复注解,使用方式如下:
不同的地方是,创建重复注解 Role 时加上了 @Repeatable 注解,指向存储注解 Roles,这样在使用时就可以直接重复使用 Role 注解。从上面例子看出,使用 @Repeatable 注解更符常规思维,可读性强一点。public @interface Roles {
Role[] value();
}
@Repeatable(Roles.class)
public @interface Role {
String roleName();
}
public class RoleTest {
@Role(roleName = "role1")
@Role(roleName = "role2")
public String doString(){
return "这是C语言中文网Java教程";
}
}
两种方法获得的效果相同。重复注解只是一种简化写法,这种简化写法是一种假象,多个重复注解其实会被作为“容器”注解的 value 成员的数组元素处理。
@Native
使用 @Native 注解修饰成员变量,则表示这个变量可以被本地代码引用,常常被代码生成工具使用。对于 @Native 注解不常使用,了解即可。- 随机文章
- 马尔代夫语言书写( )
- 马尔代夫飞燕昆明(马尔代夫航空公司开通昆明直飞航线)
- 旅游 退款 马尔代夫(修改如下:如何在马尔代夫旅游后安全退款?)
- 上海 马尔代夫 旅游(上海游客可直飞马尔代夫,畅游神仙岛!)
- 东方马尔代夫介绍(东方马尔代夫:神秘又浪漫的度假胜地)
- 嗡嗡 马尔代夫(嗡嗡声在马尔代夫的典雅风尚)
- 新航 马尔代夫(新加坡航空开通马尔代夫航线)
- 长滩 马尔代夫(长滩岛与马尔代夫 - 热带岛屿之美)
- 春节 马尔代夫(春节时光在马尔代夫沉浸式度假)
- 狮子想去马尔代夫(狮子向往马尔代夫,寻找梦幻海岛之旅)
- 迪拜马尔代夫对比(迪拜和马尔代夫:一场度假胜地的对决)
- 海航马尔代夫防疫(海航积极应对疫情,保障马尔代夫安全)
- 许晴马尔代夫电影(许晴斩获马尔代夫电影节最佳女演员奖)
- 玉林马尔代夫海滩(玉林尽享马尔代夫风情,海滩清新宜人)
- 烟台马尔代夫别墅(烟台别墅模拟马尔代夫:避暑好去处!)
- 嵊泗 马尔代夫(从嵊泗到马尔代夫:一个旅行家的海岛之旅)
- 荆门马尔代夫在哪(荆门华瑞湖畔马尔代夫酒店位置在哪?)
- 迪拜马尔代夫位置(迪拜皇家度假村坐落于美丽的马尔代夫)
- 孟加拉踢马尔代夫(孟加拉击败马尔代夫,赢得比赛冠军。)
- 马尔代夫中国订单(中国游客订单成马尔代夫旅游市场主力)
- 电信 马尔代夫(电信进军马尔代夫市场,提供全方位通讯服务)
- 马尔代夫中国舰艇(中国海军首次在马尔代夫执行护航任务)
- 郑和马尔代夫视频(郑和船队神秘亮相、马尔代夫视频曝光)
- 马尔代夫儿童内裤(后的:马尔代夫儿童内衣问题引发关注)
- 马尔代夫内衣男模(马尔代夫男模穿上性感内衣上演时装秀)
- 马尔代夫买卡地亚(马尔代夫免税店现售卡地亚,别错过!)
- 马尔代夫分港口吗(马尔代夫港口重新分配,优化水运交通)
- 马尔代夫出现大坑(马尔代夫现现深坑,游客需注意安全!)
- 淄博 马尔代夫(淄博旅游公司推出马尔代夫旅游,预订火热进行中)
- 马尔代夫出行提示(马尔代夫旅行注意事项,安全出行指南)
