1. 什么是注解

注解是java5引入的特性,在代码中插入一种注释化的信息,用于对代码进行说明,可以对包、类、接口、字段、方法参数、局部变量等进行注解。注解也叫元数据(meta data)。这些注解信息可以在编译期使用预编译工具进行处理(pre-compiler tools),也可以在运行期使用 Java 反射机制进行处理。

它主要的作用有以下四方面:

  • 生成文档,通过代码里标识的元数据生成javadoc文档。
  • 编译检查,通过代码里标识的元数据让编译器在编译期间进行检查验证。
  • 编译时动态处理,编译时通过代码里标识的元数据动态处理,例如动态生成代码。
  • 运行时动态处理,运行时通过代码里标识的元数据动态处理,例如使用反射注入实例。

一般注解可以分为三类:

  • 一类是Java自带的标准注解,包括@Override@Deprecated@SuppressWarnings,分别用于标明重写某个方法、标明某个类或方法过时、标明要忽略的警告,用这些注解标明后编译器就会进行检查。
  • 一类为元注解,元注解是用于定义注解的注解,包括@Retention@Target@Inherited@Documented@Retention用于标明注解被保留的阶段,@Target用于标明注解使用的范围,@Inherited用于标明注解可继承,@Documented用于标明是否生成javadoc文档。
  • 一类为自定义注解,可以根据自己的需求定义注解,并可用元注解对自定义注解进行注解。

2. 注解的原理

为什么在类、方法上加一个注解,就可以通过getAnnotation()获取到申明的注解的值?

比如:

  1. @Test("test")
  2. public class AnnotationTest {
  3. public void test(){
  4. }
  5. }

对于注解test就可以在运行时通过AnnotationTest.class.getAnnotation(Test.class)获取注解声明的值。

  1. 注解信息保存在哪儿?
  2. 注解信息如何获取?

在AnnotationTest类被编译后,在对应的AnnotationTest.class文件中会包含一个RuntimeVisibleAnnotations属性,由于这个注解是作用在类上,所以此属性被添加到类的属性集上。即Test注解的键值对value=test会被记录起来。而当JVM加载AnnotationTest.class文件字节码时,就会将RuntimeVisibleAnnotations属性值保存到AnnotationTest的Class对象中,于是就可以通过AnnotationTest.class.getAnnotation(Test.class)获取到Test注解对象,进而再通过Test注解对象获取到Test里面的属性值。

这里可能会有疑问,Test注解对象是什么?其实注解被编译后的本质就是一个继承Annotation接口的接口,所以@Test其实就是“public interface Test extends Annotation”,当我们通过AnnotationTest.class.getAnnotation(Test.class)调用时,JDK会通过动态代理生成一个实现了Test接口的对象,并把将RuntimeVisibleAnnotations属性值设置进此对象中,此对象即为Test注解对象,通过它的value()方法就可以获取到注解值。

3. 四种类型注解

注解可以通过作用的类型可分为:类注解、方法注解、参数注解、变量注解。

类注解

访问类注解的例子:

  1. Class aClass = TheClass.class;
  2. Annotation[] annotations = aClass.getAnnotations();
  3. for(Annotation annotation : annotations){
  4. if(annotation instanceof MyAnnotation){
  5. MyAnnotation myAnnotation = (MyAnnotation) annotation;
  6. System.out.println("name: " + myAnnotation.name());
  7. System.out.println("value: " + myAnnotation.value());
  8. }
  9. }

此方法获取到的是该类的所有注解,也可指定某一个类:

  1. Class aClass = TheClass.class;
  2. Annotation annotation = aClass.getAnnotation(MyAnnotation.class);
  3. if(annotation instanceof MyAnnotation){
  4. MyAnnotation myAnnotation = (MyAnnotation) annotation;
  5. System.out.println("name: " + myAnnotation.name());
  6. System.out.println("value: " + myAnnotation.value());
  7. }

Spring中的@Service``@Controller即是如此。

方法注解

  1. public class TheClass {
  2. @MyAnnotation(name="someName", value = "Hello World")
  3. public void doSomething(){}
  4. }

获取方法对象的指定注解:

  1. Method method = ... // 获取方法对象
  2. Annotation annotation = method.getAnnotation(MyAnnotation.class);
  3. if(annotation instanceof MyAnnotation){
  4. MyAnnotation myAnnotation = (MyAnnotation) annotation;
  5. System.out.println("name: " + myAnnotation.name());
  6. System.out.println("value: " + myAnnotation.value());
  7. }

SpringMVC中的@RequestMapping即是如此。

参数注解

方法的参数也可以添加注解:

  1. public class TheClass {
  2. public static void doSomethingElse(
  3. @MyAnnotation(name="aName", value="aValue") String parameter){
  4. }
  5. }

访问注解:

  1. Method method = ... //获取方法对象
  2. Annotation[][] parameterAnnotations = method.getParameterAnnotations();
  3. Class[] parameterTypes = method.getParameterTypes();
  4. int i=0;
  5. for(Annotation[] annotations : parameterAnnotations){
  6. Class parameterType = parameterTypes[i++];
  7. for(Annotation annotation : annotations){
  8. if(annotation instanceof MyAnnotation){
  9. MyAnnotation myAnnotation = (MyAnnotation) annotation;
  10. System.out.println("param: " + parameterType.getName());
  11. System.out.println("name : " + myAnnotation.name());
  12. System.out.println("value: " + myAnnotation.value());
  13. }
  14. }
  15. }

Mybatis中的@Param即是如此

变量注解

使用注解:

  1. public class TheClass {
  2. @MyAnnotation(name="someName", value = "Hello World")
  3. public String myField = null;
  4. }

获取注解:

  1. Field field = ...//获取方法对象</pre>
  2. <pre>
  3. Annotation annotation = field.getAnnotation(MyAnnotation.class);
  4. if(annotation instanceof MyAnnotation){
  5. MyAnnotation myAnnotation = (MyAnnotation) annotation;
  6. System.out.println("name: " + myAnnotation.name());
  7. System.out.println("value: " + myAnnotation.value());
  8. }

Spring中的@Inject``@Resource即是如此。

参考资料:

注解机制及其原理

Java Reflection(八):注解

java中的注解总结的更多相关文章

  1. java中的注解(Annotation)

    转载:https://segmentfault.com/a/1190000007623013 简介 注解,java中提供了一种原程序中的元素关联任何信息.任何元素的途径的途径和方法. 注解是那些插入到 ...

  2. java中元注解

    java中元注解有四个: @Retention @Target @Document @Inherited:  @Retention:注解的保留位置 @Retention(RetentionPolicy ...

  3. java中的注解详解和自定义注解

    一.java中的注解详解 1.什么是注解 用一个词就可以描述注解,那就是元数据,即一种描述数据的数据.所以,可以说注解就是源代码的元数据.比如,下面这段代码: @Override public Str ...

  4. 【java】细说 JAVA中 标注 注解(annotation)

    Java注解是附加在代码中的一些元信息,用于一些工具在编译.运行时进行解析和使用,起到说明.配置的功能.注解不会也不能影响代码的实际逻辑,仅仅起到辅助性的作用 下面我们来详细说说这个注解,到底是怎么一 ...

  5. 【java】详解java中的注解(Annotation)

    目录结构: contents structure [+] 什么是注解 为什么要使用注解 基本语法 4种基本元注解 重复注解 使用注解 运行时处理的注解 编译时处理的注解 1.什么是注解 用一个词就可以 ...

  6. Java中的注解基础

    一.元注解 元注解的作用就是负责注解其他注解. 1.@Target @Target用来指明注解所修饰的目标,包括packages.types(类.接口.枚举.Annotation类型).类型成员(方法 ...

  7. 【转载:java】详解java中的注解(Annotation)

    目录结构: contents structure [+] 什么是注解 为什么要使用注解 基本语法 4种基本元注解 重复注解 使用注解 运行时处理的注解 编译时处理的注解 1.什么是注解 用一个词就可以 ...

  8. Java中的注解及自定义注解你用的怎么样,能不能像我这样应用自如?

    Java注解提供了关于代码的一些信息,但并不直接作用于它所注解的代码内容.在这个教程当中,我们将学习Java的注解,如何定制注解,注解的使用以及如何通过反射解析注解. Java1.5引入了注解,当前许 ...

  9. 注解式项目开发!详细解析Java中各个注解的作用和使用方式

    @Target 作用: 指明了修饰的这个注解的使用范围, 即被描述的注解可以用在哪里 @Target(ElementType.Type) ElementType取值的类型: TYPE: 类,接口或者枚 ...

随机推荐

  1. vue组件(将页面公用的头部组件化)

    呃……重新捡起前面用vue-cli快速生成的项目. 之前是做过一个多页面的改造,以及引入vux的ui组件,这次在这个项目的基础上,再来聊聊vue中的component. 别问我为啥总是写关于vue的博 ...

  2. css浮动布局,浮动原理,清除(闭合)浮动方法

    css浮动 1.什么是浮动:在我们布局的时用到的一种技术,能够方便我们进行布局,通过让元素浮动,我们可以使元素在水平上左右移动,再通过margin属性调整位置 2.浮动的原理:使当前元素脱离普通流,相 ...

  3. 【Netty】Netty核心组件介绍

    一.前言 前篇博文体验了Netty的第一个示例,下面接着学习Netty的组件和其设计. 二.核心组件 2.1. Channel.EventLoop和ChannelFuture Netty中的核心组件包 ...

  4. Mysql存储过程使用LEAVE实现MSSQL存储过程中return语法

    DELIMITER $$ USE `qrsoft_dyj_db`$$ DROP PROCEDURE IF EXISTS `proc_withdraw_approve`$$ CREATE PROCEDU ...

  5. 解决U盘容量变小问题

    今天又想重新给电脑刷刷kali linux新版本了貌似N久没更,直接重新刷系统吧...然后发现USB容量变小,这就尴尬了,接着总结了个小方法. 解决方法:1.先把u盘插好,运行cmd,2.输入disk ...

  6. selenium基础框架的封装(Python版)

    一.常用函数的封装 在使用selenium做web自动化测试的过程中,经常会碰到各种各样的问题,比如: 1.页面加载比较慢时,selenium查找元素抛出异常,导致脚本运行中止 2.写完脚本后发现代码 ...

  7. python selenium 元素定位(三)

    上两篇的博文中介绍了python selenium的环境搭建和编写的第一个自动化测试脚本,从第二篇的例子中看出来再做UI级别的自动化测试的时候,有一个至关重要的因素,那就是元素的定位,只有从页面上找到 ...

  8. 020 <one-to-one>、<many-to-one>单端关联上的lazy(懒加载)属性

    <one-to-one>.<many-to-one>单端关联上,可以取值:false/proxy/noproxy(false/代理/不代理) 实例一:所有lazy属性默认(支持 ...

  9. Chart.js – 效果精美的 HTML5 Canvas 图表库

    Chart.js 是一个令人印象深刻的 JavaScript 图表库,建立在 HTML5 Canvas 基础上.目前,它支持6种图表类型(折线图,条形图,雷达图,饼图,柱状图和极地区域区).而且,这是 ...

  10. 重构了cxlt-vue2-toastr插件

    距离上篇文章已经过去一个多月了,期间有很多想法,但时间真不是想挤就能挤出来的.其实这段时间我就做了一件事,一个小程序的项目,已上线半月有余,也迭代了几个版本,现在还在不断完善. 先说点题外话,我们做了 ...