JDK1.5开始增加了Annotation功能,该功能可用于:

1、类;

2、构造方法;

3、成员变量;

4、方法

5、参数

等的声明;

该功能并不影响程序的运行,但是会对编译器警告等辅助工具产生影响。本文将介绍Annotation功能的使用方法。


1、定义Annotation类型

在定义Annotation类型时,也需要用到来定义接口的interface关键字,但需要在interface关键字前加一个@符号,即表示定义Annotation类型的关键字为@interface,这个关键字的隐含意思是继承了java.lang.annotation.Annotation接口。例如,下面的代码就定义了一个Annotation类型。

public @interface NoMemberAnnotation{}

上面定义的Annotation类型@NoMemberAnnotation未包含任何成员,这样的Annotation类型被称为marker annotation。下面的代码定义了一个只包含一个成员的Annotation类型。

public @interface OneMemberAnnotatino{
String value();
}

String: 成员类型。可用的成员类型有:

1)String;

2)Class;

3)基本类型;

4)枚举;

5)Annotation;

value:成员的名称。如果在所定义的Annotation类型中只包含一个成员,通常将成员命名为value;

下面代码定义了一个包含多个成员的Annotation类型:

public @interface MoreMemberAnnotation{
String describe();
Class type();
}

在为Annotation类型定义成员时,也可以为成员设置默认值。例如,下面的代码在定义Annotation类型时就为成员设置了默认值。

public @interface DefaultValueAnnotation{
String describe() default "<默认值>";
Class type(0 default void.class;
}

在定义Annotation类型时,还可以通过Annotation类型@Target来设置Annotation类型使用的程序元素种类。如果未设置@Target,则表示适用于所有程序元素。枚举类ElementType中的枚举常量用来设置@Target,如下:

  • ANNOTATION_TYPE :表示用于Annotation类型
  • TYPE : 表示用于类、接口和枚举,以及Annotation类型
  • CONSTRUCTOR : 表示用于构造方法
  • FIELD : 表示用于成员变量和枚举常量
  • METHOD :表示用于方法
  • PARAMETER :表示用于参数
  • LOCAL_VARIABLE :表示用于局部变量
  • PACKAGE :表示用于包

通过Annotation类型的@Retention可以设置Annotation的有效范围。枚举类RetentionPolicy中的枚举常量用于设置@Retention,如果没有设置,那么默认为CLASS常量的范围,如下所示:

  • SOURCE :表示不编译Annotation到类文件中,有效范围最小
  • CLASS :表示编译Annotation到类文件中,但是在运行时不加载Annotation到JVM中
  • RUNTIME :表示在运行时加载Annotation到JVM中,有效范围最大

2、访问Annotation信息

如果在定义Annotation类型时将@Retention设置为RetetionPolicy.RUNTIME,那么在运行程序时通过反射就可以获取到相关的Annotation信息,如获取构造方法、字段和方法的Annotation信息。

类Constructor、Field和Method继承了AccessibleObject类,而AccessibleObject类实现了AnnotatedElement,因此也就是说,Constructor、FIeld、Method均可以使用AccessibleObject实现的方法来操作注解。(注:这里Class类直接实现了AnnotatedElement不是继承AccessibleObject)

下面,我们看看AccessibleObject继承来的方法:

<T extends Annotation> getAnnotation(Class<T> annotationClass);

如果该元素存在annotationClass类型的注释,则返回该注释,否则返回null;

Annotation[] getAnnotations()

返回该元素的所有注解

Annotation[] getDeclaredAnnotation()

返回直接存在于该元素的注解(例如继承而来的注解不会被返回)

boolean isAccessible()

该元素是否执行安全检查(安全检查如:变量设置了private属性不可以直接访问,设置为取消检查则可以反问该元素)

boolean isAnnotationPresent(Class<? extends Annotation> annotationClass)

该元素是否存在annotationClass注解

static void setAccessible(AccessibleObject[] array, boolean flag)

为一组元素设置是否执行安全检查

void setAccessible(boolean flag)

为该元素设置是否执行安全检查


3、DEMO

package cn.lay.annotation;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; /**
* annotation demo
* @ClassName: AnnotationDemo
* @Description:
* @author: lay
* @date: Mar 18, 2018
*/
@Test(
value = "test",
ages = {23, 23, 34, 35, 34},
bookEnum = BookEnum.JAVA,
cls = String.class,
test2s = {@Test2(), @Test2("lay"), @Test2("zhang"), @Test2("wu")}
)
public class AnnotationDemo {
public static void main(String[] args) {
// 判断是否注解了该类型
if (AnnotationDemo.class.isAnnotationPresent(Test.class)) {
// 获取该注解
Test annotation = AnnotationDemo.class.getAnnotation(Test.class);
// 取值
String value = annotation.value();
// 输出
System.out.println("value=" + value);
int[] ages = annotation.ages();
for (int age : ages) {
System.out.println("age=" + age);
}
BookEnum bookEnum = annotation.bookEnum();
System.out.println("bookName=" + bookEnum.getBookName());;
Class<?> cls = annotation.cls();
System.out.println("class=" + cls.getName());
Test2[] test2s = annotation.test2s();
for (Test2 test2 : test2s) {
System.out.println("test2=" + test2.value());
}
}
}
} /**
* 定义注解
* @ClassName: Test
* @Description:
* @author: lay
* @date: Mar 18, 2018
*/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface Test{
String value();
int[] ages();
BookEnum bookEnum();
Class<?> cls();
Test2[] test2s();
} @Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface Test2{
String value() default "lily";
} enum BookEnum{ JAVA("java"),
SQL("sql"); private String bookName; public String getBookName() {
return bookName;
} public void setBookName(String bookName) {
this.bookName = bookName;
} private BookEnum(String bookName) {
this.bookName = bookName;
}
}

控制台打印:

value=test
age=23
age=23
age=34
age=35
age=34
bookName=java
class=java.lang.String
test2=lily
test2=lay
test2=zhang
test2=wu

更多详细内容,参阅:http://tool.oschina.net/apidocs/apidoc?api=jdk-zh

Java API 之 Annotation功能的更多相关文章

  1. ElasticSearch7.3学习(三十)----ES7.X SQL新特性解析及使用Java api实现sql功能

    一.ES7 sql新特性 1.1 数据准备 创建索引及映射 建立价格.颜色.品牌.售卖日期 四个字段 PUT /tvs PUT /tvs/_mapping { "properties&quo ...

  2. 第08章 ElasticSearch Java API

    本章内容 使用客户端对象(client object)连接到本地或远程ElasticSearch集群. 逐条或批量索引文档. 更新文档内容. 使用各种ElasticSearch支持的查询方式. 处理E ...

  3. 【转】Java奇技淫巧-插件化注解处理API(Pluggable Annotation Processing API)

    参考资料 JDK6的新特性之六:插入式注解处理API(Pluggable Annotation Processing API) Java Annotation Processing and Creat ...

  4. 常用Java API之Scanner:功能与使用方法

    Scanner 常用Java API之Scanner:功能与使用方法 Scanner类的功能:可以实现键盘输入数据到程序当中. 引用类型的一般使用步骤:(Scanner是引用类型的) 1.导包 imp ...

  5. API 注解 & Java API annotation

    API 注解 & Java API annotation 注解 annotation

  6. Java基础笔记 – Annotation注解的介绍和使用 自定义注解

    Java基础笔记 – Annotation注解的介绍和使用 自定义注解 本文由arthinking发表于5年前 | Java基础 | 评论数 7 |  被围观 25,969 views+ 1.Anno ...

  7. RabbitMQ的Java API编程

    1.创建Maven工程,pom.xml引入依赖: <dependency> <groupId>com.rabbitmq</groupId> <artifact ...

  8. Java API 各个包的内容解释

    java.applet 提供创建 applet 所必需的类和 applet 用来与其 applet 上下文通信的类. java.awt 包含用于创建用户界面和绘制图形图像的所有类. java.awt. ...

  9. 插件化注解处理API(Pluggable Annotation Processing API)

    Java奇技淫巧-插件化注解处理API(Pluggable Annotation Processing API) 参考资料 JDK6的新特性之六:插入式注解处理API(Pluggable Annota ...

随机推荐

  1. [agc004d]Teleporter 暴力

    Description 维尼管辖的领土很大,我们可以抽象为n个城市,其中1号点为首都.这n个城市之有n条单向电缆,一条信息经过一条电缆进行传输所需时间会+1s,然而维尼并不能忍受时间白白被续,他要求从 ...

  2. web渗透-sql注入

    何为sql注入 所谓SQL注入,就是通过把SQL命令插入到Web表单提交或输入域名或页面请求的查询字符串,最终达到欺骗服务器执行恶意的SQL命令,得到想要得到的信息. OWASPTop 10 此处的注 ...

  3. 各种 Python 库/模块/工具

    1 算法 1.1 字符串处理 re 正则表达式的标准库. StringIO / cStringIO 以读写文件的方式来操作字符串(有点类似于内存文件). cStringIO 是 C 语言实现的,提供高 ...

  4. Bootrap 项目实战(微金所前端首页)第一部分

    微金所前端首页成果图:(这是本人自己按照微金所官网首页,采用Bootrap,JS,JQuery,css制作的网页效果图,在第二部分我会公布网页源代码) 如需网页源代码,请在下方留言,备注你的qq邮箱. ...

  5. image_size must contain 3 elements[4]

    今天在制作自己训练集合并且训练的时候,碰见了如下的错: image_size must contain 3 elements[4] 这是因为训练的数据集中不是所有的图片位深都是三通道的. 写一个脚本查 ...

  6. 数据库--sql文件

    sql 脚本是包含一到多个 sql 命令的 sql 语句集合 使用 Linux: mysqldump 命令 1.导出数据和表结构: mysqldump -u 用户名 -p 数据库名称 > nam ...

  7. JAVA中 package 和 import 的使用

    1.打包--package 包名一般为小写,而类名的第一个字母一般为大写,这样在引用时,可以明显的分辨出包名和类名.如果在类的定义之前没有使用package定义包名,那么该类就属于缺 省的包. 1.1 ...

  8. 数据库SQL(1)

    EG1:db.LpOutputGroups.GroupBy(q => q.CalcGroupDesc).ToList().OrderByDescending(m => m.First(). ...

  9. SpringBoot集成WebSocket【基于纯H5】进行点对点[一对一]和广播[一对多]实时推送

    代码全部复制,仅供自己学习用 1.环境搭建 因为在上一篇基于STOMP协议实现的WebSocket里已经有大概介绍过Web的基本情况了,所以在这篇就不多说了,我们直接进入正题吧,在SpringBoot ...

  10. J15W-J45W黄铜截止阀厂家,J15W-J45W黄铜截止阀价格 - 专题栏目 - 无极资讯网

    无极资讯网 首页 最新资讯 最新图集 最新标签   搜索 J15W-J45W黄铜截止阀 无极资讯网精心为您挑选了(J15W-J45W黄铜截止阀)信息,其中包含了(J15W-J45W黄铜截止阀)厂家,( ...