Annotation(注解)简单介绍:

注解大家印象最深刻的可能就是JUnit做单元測试,和各种框架里的使用了。

本文主要简介一下注解的用法,下篇文章再深入的研究。

annotation并不直接影响代码语义。可是它可以被看作类似程序的工具或者类库。它会反过来对正在执行的程序语义有所影响。

annotation能够从源文件,class文件或者以在执行时反射的多种方式被读取

java注解系统自带有主要下面几个注解:

Override注解表示子类要重写(override)父类的相应方法

Deprecated注解表示方法是不建议被使用的

Suppress Warnings注解表示抑制警告

怎样自己定义注解:

仅仅须要使用@interface来定义一个注解,比如:

//使用@interface来声明一个注解(实际上是自己主动继承了java.lang.annotation.Annotation接口)
public @interface AnnotationTest {
String value1() default "hello"; //为注解设置String类型的属性Value1,并使用defalutkeyword设置默认值
EnumTest value2(); //设置枚举类型的value2
String[] value3(); //设置数组类型的value3
}

怎样来使用注解呢,例如以下:

@AnnotationTest(value2 = EnumTest.age, value3={""})
public class AnnotationUsage { @AnnotationTest(value1 = "Test", value2 = EnumTest.name, value3={""})
String test; @AnnotationTest(value1 = "Test", value2 = EnumTest.name, value3={""})
public void method(){
System.out.println("usage of Annotation");
}
}

如上,注解能够标注在属性。方法。类上。

须要使用name=value这样的赋值方式为属性赋值,由于value1设置了默认属性,所以能够忽略,假设没有设置默认值则全部的属性都要一一赋值。

另一种特殊情况,假设注解里仅仅定义了一个属性,名字是value,那么能够直接赋值,不须要使用name=value这样的赋值方式。例如以下:

public @interface AnnotationTest {
String value();
} @AnnotationTest("test")
public void method(){
System.out.println("usage of Annotation");
}

修饰注解的“注解”

注解也能够加入注解的“注解”去修饰,经常使用的有下面两个,一个是Retention。一个Target

Retention:

使用Retention必需要提供一个为java.lang.annotation.RetentionPolicy类型的的枚举

RetentionPolicy枚举有下面3个类型:

SOURCE:编译程序时处理完Annotation信息后就完毕任务

CLASS:编译程序将Annotation存储于class文件里,不能够由虚拟机读入

RUNTIME:编译程序将Annotation存储于class文件里。能够由虚拟机读入

用这三种Retention的Prolicy能够决定注解是从源文件。class文件或者以在执行时反射被读取

关于Retention的样例在最后

Target:

使用java.lang.annotation.Target能够定义注解被使用的位置

相同,在使用时要指定一个java.lang.annotation.ElementType的枚举值类型为他的“属性”

ElementType枚举的类型例如以下:

ANNOTATION_TYPE:适用于annotation

CONSTRUCTOR:适用于构造方法

FIELD:适用于field

LOCAL_VARIABLE:适用于局部变量

METHOD:适用于方法

PACKAGE:适用于package

PARAMETER:适用于method上的parameter

TYPE:适用于class,interface,enum

例如以下:定义一个注解MyTarget。设置Target类型为Method来修饰这个注解。这样这个注解仅仅能标注在method的方法上

@Target(ElementType.METHOD)
public @interface MyTarget {
String hello() default "hello";
}
@MyTarget  //这里则会报错,由于他标注在类上面了
public class MyTargetTest {
@MyTarget //标注在方法上不会报错
public void doSomething(){
System.out.println("hello world");
}
}


使用反射调用注解

在下面的类中Class Constructor Field Method Package等类都实现了AnnotatedElement接口
在接口中有下面重要的方法:

getAnnotations(Class annotationType)获取一个指定的annotation类型

getAnnotations() 获取全部的Annotation

getDeclaredAnnotations() 获取声明过的全部Annotation

isAnnotationPresent(Class<? extends Annotation> annotationClass)这个annotation是否出现


通过这些方法,配合反射我们就能够在程序执行时拿到注解的内容了。样例例如以下:
@Retention(RetentionPolicy.RUNTIME)	//定义一个注解。使用Retention标注为RUNTIME
public @interface MyAnnotation {
String hello() default "hello";
String world();
}

该注解被标示为runtime类型,表示该注解最后能够保存在class文件里,并为java虚拟机在执行时读取到

@Retention(RetentionPolicy.CLASS)	//定义一个注解。Retention标注为RUNTIME
public @interface MyAnnotation2 {
String hello() default "hello"; //设置默认值为hello
}

自己定义的还有一个注解Retention标示为class

public class MyTest {
@SuppressWarnings("unchecked") //java自带的注解Retention的policy为SOURCE
@Deprecated //java自带的注解Retention的policy为RUNTIME
@MyAnnotation(Name="Dean", Age="25") //自己定义的注解Retention的policy为RUNTIME
@MyAnnotation2 //自己定义的注解Retention的policy为CLASS
public void TestMethod() {
System.out.println("this is a method");
}
}

定义一个TestMethod方法。给他标示了4个注解。当中2个java自带的,2个我们自己定义的。注解的的Retention属性各不同样。

以下定义一个測试类来验证结果:

public static void main(String[] args) throws Exception {

		MyTest myTest = new MyTest();
//通过反射得到TestMethod方法
Class<MyTest> c = MyTest.class;
Method method = c.getMethod("TestMethod", new Class[]{}); //AnnotatedElement接口中的方法isAnnotationPresent(),推断传入的注解类型是否存在
if (method.isAnnotationPresent(MyAnnotation.class)) {
method.invoke(myTest, new Object[]{});
//AnnotatedElement接口中的方法getAnnotation(),获取传入注解类型的注解
MyAnnotation myAnnotation = method.getAnnotation(MyAnnotation.class);
//拿到注解中的属性
String name = myAnnotation.Name();
String age = myAnnotation.Age();
System.out.println("name:"+name +" age:"+age);
} System.out.println("-----------------------------------"); //AnnotatedElement接口中的方法getAnnotations(),获取全部注解
Annotation[] annotations = method.getAnnotations();
//循环注解数组打印出注解类型的名字
for (Annotation annotation : annotations) {
System.out.println(annotation.annotationType().getName());
}
}

打印结果为:

this is a method

name:Dean   age:25

-----------------------------------

java.lang.Deprecated

gxy.text.annotation.MyAnnotation

切割线上:介绍了怎样使用AnnotatedElement接口中的方法和反射去调用注解

切割线下:证明了仅仅有定义了Retention的Policy为Runtime的注解才干够被反射读取出来

下一篇文章分析一下在JUnit中反射与注解的使用和原理

Java反射学习总结五(Annotation(注解)-基础篇)的更多相关文章

  1. java web 学习十五(jsp基础语法)

    任何语言都有自己的语法,JAVA中有,JSP虽然是在JAVA上的一种应用,但是依然有其自己扩充的语法,而且在JSP中,所有的JAVA语句都可以使用. 一.JSP模版元素 JSP页面中的HTML内容称之 ...

  2. Java反射学习总结终(使用反射和注解模拟JUnit单元测试框架)

    转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持! 本文是Java反射学习总结系列的最后一篇了,这里贴出之前文章的链接,有兴趣的可以打开看看. ...

  3. Java反射学习总结四(动态代理使用实例和内部原理解析)

    通过上一篇文章介绍的静态代理Java反射学习总结三(静态代理)中,大家可以发现在静态代理中每一个代理类只能为一个接口服务,这样一来必然会产生过多的代理,而且对于每个实例,如果需要添加不同代理就要去添加 ...

  4. Java反射学习系列-绪论

    Java反射学习系列-绪论 https://blog.csdn.net/hanchao5272/article/details/79358924

  5. Java反射理解(五)-- 方法反射的基本操作

    Java反射理解(五)-- 方法反射的基本操作 方法的反射 1. 如何获取某个方法 方法的名称和方法的参数列表才能唯一决定某个方法 2. 方法反射的操作 method.invoke(对象,参数列表) ...

  6. Java IO学习笔记四:Socket基础

    作者:Grey 原文地址:Java IO学习笔记四:Socket基础 准备两个Linux实例(安装好jdk1.8),我准备的两个实例的ip地址分别为: io1实例:192.168.205.138 io ...

  7. Java IO学习笔记五:BIO到NIO

    作者:Grey 原文地址: Java IO学习笔记五:BIO到NIO 准备环境 准备一个CentOS7的Linux实例: 实例的IP: 192.168.205.138 我们这次实验的目的就是直观感受一 ...

  8. Android开发学习之路--Annotation注解简化view控件之初体验

    一般我们在写android Activity的时候总是会在onCreate方法中加上setContentView方法来加载layout,通过findViewById来实现控件的绑定,每次写这么多代码总 ...

  9. Java反射学习:深入学习Java反射机制

    一.Java反射的理解(反射是研究框架的基础之一) Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的 ...

随机推荐

  1. BZOJ1565 [NOI2009]植物大战僵尸 【最大权闭合子图 + tarjan缩点(或拓扑)】

    题目 输入格式 输出格式 仅包含一个整数,表示可以获得的最大能源收入.注意,你也可以选择不进行任何攻击,这样能源收入为0. 输入样例 3 2 10 0 20 0 -10 0 -5 1 0 0 100 ...

  2. Python学习笔记(Django篇)——1、环境搭建篇(如何在Pycharm中配置Python和Django)

      1.准备好以下东东,并且按照先后顺序进行安装: Python 3.6 (64-bit) Django-1.11.tar.gz pycharm-community-2016.3.2.exe 安装好了 ...

  3. 【ZOJ4070】Function and Function(签到)

    题意:求 k 层嵌套的 f(x) 0<=x,k<=1e9 思路:迭代不会很多次后函数里就会=0或者1,再看层数奇偶直接返回答案 #include<cstdio> #includ ...

  4. css3文件树

    <!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8" ...

  5. 培训补坑(day3:网络流&最小割)

    继续补坑.. 第三天主要是网络流 首先我们先了解一下网络流的最基本的算法:dinic 这个算法的主要做法就是这样的: 在建好的网络流的图上从源点开始向汇点跑一遍BFS,然后如果一条边的流量不为0,那么 ...

  6. android hook 框架 ADBI 简介、编译、运行

    Android so注入-libinject2 简介.编译.运行 Android so注入-libinject2  如何实现so注入 Android so注入-Libinject 如何实现so注入 A ...

  7. 判断dataset是否被修改—DataSet.HasChanges 方法

    DataSet.HasChanges 方法 获取一个值,该值指示 DataSet 是否有更改,包括新增行.已删除的行或已修改的行. 命名空间:   System.Data程序集:  System.Da ...

  8. 找出数字数组中最大的元素(使用Math.max函数)

    从汤姆大叔的博客里看到了6个基础题目:本篇是第1题 - 找出数字数组中最大的元素(使用Match.max函数) 从要求上来看,不能将数组sort.不能遍历.只能使用Math.max,所以只能从java ...

  9. python获取小程序手机号并绑定

    最近在做小程序开发,在其中也遇到了很多的坑,获取小程序的手机号并绑定就遇到了一个很傻的坑. 流程介绍 官方流程图 小程序使用方法 需要将 <button> 组件 open-type 的值设 ...

  10. (转)代码中实现button

    链接地址:http://www.cnblogs.com/hukezhu/p/4500206.html 随着iOS开发发展至今,在UI制作上逐渐分化为了三种主要流派:使用代码手写UI及布局:使用单个xi ...