写点什么

9000 字通俗易懂的讲解下 Java 注解,你还看不明白?,linux 操作系统基础与应用答案

用户头像
极客good
关注
发布于: 刚刚
  1. 自定义注解(实际很少)

  2. JDK 内置注解(@Override 检验方法重写)

  3. 框架中的注解


不知道这个能不能理解,就是说,对于注解而言,是有几种不同分类的,首先,我们可以自己写一个注解出来(下面会讲),另外对于 JDK 本身而言有自己的的注解,我们看个代码,你就知道了:



比如这个,是重写 toString 方法,上面就有个 JDK 的内置注解 @Override,这个注解就起到一个检验的作用,因为它是 Object 的方法,你现在要重写它,那么名字啊,参数啊要和之前的一样,不一样,就给你报错,不信你试试:



这个是关于 JDK 的内置注解,那么最后一个关于框架的注解,我想你只要学过 Spring 都知道,比如 @Controller,熟悉吧,这就是框架中的注解。

注解的本质

经过上面的讲解,我们应该大致了解了什么是注解,以及注解的一些分类,现在,我们对于概念上的注解算是清楚了,但是这个注解本质是个什么呢?


告诉你吧,注解的本质是个接口,为啥,先来看下,如何定义一个注解(下面会详细讲解)


public @interface Main {}


就这些,就定义了一个注解,不知道你发现了没,这个和接口很像啊,有啥区别,就是多了一个 @,不然就是接口啊,接下来我们使用 XJad 把这个注解反编译一下看看:



看到没,这里的 Main 直接就是 interface 定义,然后还继承了 Annotation,这个足以说明,注解其实就是接口啊。


这个暂且聊到这,记住即可!

如何定义注解

接下来我们就来聊聊如何去自定义一个注解,我们在上面说过,注解的本质其实就是接口,上面也简单演示了一个注解的定义,如下:


public @interface Main {}


想一下,我们平常怎么定义一个接口,是不是使用关键字 interface,那么类呢?是不是使用 class 关键字,也就是说啊,定义这些一般都是需要一个关键字来加以声明的,显而易见,定义注解的关键字就是 @interface,它和接口的定义就是多了一个 @,但是注解的定义却不仅仅是如此!

元注解

这里要引入一个元注解的概念,我们先来想一下,注解我们上面说了,一般可以用来标记类,接口或者方法等,那么这里就有一个问题了,比如我定义了这么一个 Main 注解:


public @interface Main {}


那么,我这个注解是不是可以用在类上,也可以用在接口或者方法上?一般类呀,接口啊,方法啊等等它们还是有点差别的,所以对于这些最好有区分,也就是说,有些注解只能标记类,有些注解只能标记方法等,这样一来就需要对注解的作用域去进行限制。


那么这个该怎么搞,答案就是元注解,那什么是元注解呢?


元注解就是标记注解的注解


啥意思,来看下,比如我们定义的这个 Main 注解,我们规定它只能用来标记方法,那么可以这样做:



我们在上面加了一个注解 @Target,后面还有参数(下面会讲),这个参数 ElementType.METHOD 就代表我们这个注解是用于注解方法的,来,试一下:



你看,可以用在我们的 main 方法上,那么是不是不能用于类呢?我们试下:



报错了,看来是不行,所以这个 @Target 就是一个元注解,可以用来注解注解,也就是标记注解的注解。


关于元注解,一般有以下主要的几个:


  1. @Documented 用于制作文档

  2. @Target 指定注解的使用位置,不指定的话任何位置都可以使用

  3. @Retention(注解的保留策略)


这里单独提一下最后一个也就是声明注解的保留策略 @Retention,这个是什么意思呢?


这个保留策略啊,简单来讲就是说你这个注解可以在哪个时间段起作用,这个就得说说我们的代码从写出来,然后编译到执行的主要三个阶段了,画个图就是这样的:



这个我已经画的很清楚了吧,一般来说,我们的注解都是要保留到运行期间的,所以一般就是这样:



当然,具体情况具体对待。


到这里你可能发现,这个注解里面可以有参数?当然是可以的,我这里简单演示下,下面讲到注解的语法的时候你就知道了:



然后再看下使用:



其实还是蛮简单的!

注解的基本使用语法

接下来我们就来看看注解的语法吧,就是注解具体是如何使用的。


对于注解,我们知道了如何去定义它,比如简单定义一个注解:



这很简单,我们继续去看,对于注解还可以定义属性:



虽然这个属性看起来很像方法,但是人家就是属性,注解还是比较特殊的,那么现在我们来使用下这个注解:



这个时候它会报错,告诉我们需要一个 value 值,其实也好理解,你的注解定义中定义的有一个 value 属性,那么你在使用的时候就需要把这个属性值给用上,那你说我可不可以不用,可以的,那定义注解属性的时候就需要给属性添加默认值,就是这样:



可以设置成一个空字符串也可以设置成具体的值。除此之外我们还可以设置多个属性值,像这样:



这里就有知识点了,如果你在使用的时候只是给一个属性值赋值,那么在使用的时候可以这样:



那有人可能疑问,我这个 hello 对应的是 value 还是 name 啊,默认对应的都是 value,所以这个要牢记。


但是给多个属性值赋值的时候就必须指明具体的属性名称了,就是这样:



PS:通过上面的介绍我们会发现注解一个比较奇怪的地方,就是对于注解而言,我们可以定义属性,但是注解的属性长得真的像方法,但是在注解里面,它就是属性,就可以直接赋值,这里需要注意下!

属性的类型

上面简单介绍了注解的属性,那么这些属性都是可以取哪些类型值呢?大致有如下这么多:


  1. 基本数据类型

  2. String

  3. 枚举

  4. Class

  5. 注解类型

  6. 数组(以上类型的一维数组)


关于数组的看个例子,比如这样:



使用的时候也是同样的道理:


如何真正的理解注解

我们平常对于注解之所以忽视的原因在于,很多地方只需要我们去使用,比如这样:



至于注解是怎么定义的以及注解是怎么起作用的都不太了解,好像需要我们自定义注解的也都很少,所以不去系统化的学习注解的话,会忽略掉注解的很多东西,只会使用,也就是 @XXX


那么,从今天开始,我希望你能够记住,对于注解而言,它一定有如下三个流程:


  1. 定义注解

  2. 使用注解

  3. 读取并执行相应流程


下面我们就以 @Repo


【一线大厂Java面试题解析+核心总结学习笔记+最新架构讲解视频+实战项目源码讲义】
浏览器打开:qq.cn.hn/FTf 免费领取
复制代码


sitory 这个注解来看看这三个流程,首先是定义注解,这个我们可以在 IDEA 中按住 Ctrl 点进去 @Repository 来看,是这样的:



](http://www.ithuangqing.vip/wp-content/uploads/2020/07/wp_editor_md_4185e9864c1c7fc41fb1173d6526a9df.jpg)


这个就是 @Repository 注解的定义,接着我们看看 @Repository 的使用:



然后就是对注解的读取了,怎么读取呢?很多人对这块是比较模糊的,这也是对注解理解最大的障碍所在。


我们一般就是使用注解,对于注解的定义和读取这块一般都是框架什么的给我们搞定了,我们不看源码一般不知道是怎么回事的,也就不清楚注解到底是怎么运行起来的,简单的理解就是注解需要靠反射去读取,然后做相应的处理。


但是我想你一定和我一样好奇,为啥加了个 @Repository 注解之后,这个 UserBean 就被装载进 Sring 容器中生成了一个 bean 呢?


还记得我在最开始就一直在说的吗?注解是需要有专门的程序取读取的,然后根绝读取到的注解获取的信息去执行相应的操作。


所以这里,在 Spring 源码中,一定有某个或者某些程序在做这个事情。

注解的读取(注解如何起作用)

上面说了注解的定义何使用,在这里单独把注解的读取拿出来说下,因为这点事理解注解的重点,很多人觉得对注解不理解的一个原因就在于不清楚加了个注解之后到底干了啥?


也就是注解到底是如何起作用的?搞明白这个,将对你理解注解有极大的帮助。

注解主要被反射读取

对于注解的读取,一般就是通过反射技术来实现,这里就有知识点了,对于反射而言,它只能读取内存中的字节码信息,然后还记得之前我们说的注解的作用域 @Target 吗?


它里面有几个主要的作用域,也就是这张图片,再来回顾下:



对于 RetentionPolicy.CLASS 而言,这个就是指的字节码这一阶段,这个时候这个字节码文件是由 Java 源文件通过 javac 编译生成,这个时候 class 字节码文件其实还是在磁盘内,并没有进入内存中。


而反射只能取读取内存中的字节码信息,所以注解的保留策略也就是这个 @Target 只能是 RUNTIME,也即运行的时候仍然可以读取。

我的理解(精华)

很多人对注解不理解,或者觉得很模糊的一个原因就是你让我定义一个注解,我也能按照基本的注解语法去定义一个注解,你说怎么使用注解我也知道在类,方法等上面使用 @+注解名称的方式,但是也就到此为主了,更进一步的理解就有点模糊了,比如:


  1. 为什么要这样用?

  2. 原理是什么,怎么起作用的?


你想啊,我们就这样在类或者方法上面写了这么一个 @+注解名称就行了?后续是怎么起作用的呢?这里你得首先清楚,注解有三大步骤:


  1. 定义注解

  2. 使用注解

  3. 读取注解(这块是大部分人缺少的,也是大部分人对注解不理解的关键所在)


再理解下什么是注解,与注释一字之差,肯定有相似之处,两者都是提供额外信息的,好比备注,注释是给我们程序员看到,看到注释我们知道某个类是干啥的,有啥用,看到方法的注释,我们知道这个方法有什么作用需要什么参数以及参数的含义等等,那么注解嘞,注解其实是给程序看到,当程序读到注解,会从注解这里得到一些信息,知道该如何处理被该注解标记的类或方法等。


好好理解上面的,我们下面再以 Spring 的一个例子来加以说明。


对于 Spring 简单的大家都知道 IOC 吧,直白点就是不用你 new 对象,需要什么直接从 Spring 容器中获取,那么首先就需要把我们的 bean 注册到 Spring 容器中吧,这个一般有 xml 配置方式和注解方式,当然我们这里要说的是注解方式,也就是使用 @+注解名称的形式,举个简单的例子,如下:



这个注解熟悉吧,它就是可以把我们的 Person 类注册到 Spring 容器中去,当然,这里就是在对这个注解的使用,我们点进去看看这个注解是怎么定义的:



这个定义我们应该已经熟悉了,对于 @Component 也是一个注解,它其实是最基础的把类注册到 Spring 容器中的注解,后来的像我们现在说的 @Repositoy 以及 @Service 和 @Controller 这些都是在 @Component 的基础上发而来。


这里就需要注意了,其实这几个注解不管是哪个,都要清楚明白的一点就是,要它们啥用,之所以需要这些注解,就是希望在哪个类上使用这些注解,就自动把这个类注册到 Spring 容器中,这个要比我们写 xml 配置简单的多,我们就在一个类上写个 @Repositoy,它就被注册到 Spring 容器中了?


是不是很神奇,然后看下注解的定义,也很简单,没啥东西啊,怎么就自动注册到 Spring 容器中了呢?


还记得之前说的注解三大步骤嘛?首先你需要定义一个注解,然后就是使用注解,那么注解是怎么起作用的就需要有程序去读注解,这个注解就好比一个标志,一个标签一样,比如这里的 @Repositoy,当一个类被这个注解标志,那么当特有的程序去读到这个注解的时候,这个程序就知道,哦,原来是要把这类注册到 Spring 容器中啊,那么程序怎么知道要把这个类注册到 Spring 容器中呢?这就是 @Repositoy 告诉它的。另外我们知道注解一般可以设置一个 value 属性值,可以通过反射技术拿到之类的,那么在具体的将这个类注册到 Spring 容器的过程中可能就会用到这个 value 属性值,比如设置成 bean 的名字。


我们一般使用了注解,在 Spring 配置文件中就需要配置注解的包扫描:


<context:component-scan base-package="com.ithuangqing.*"/>


这个其实就是在扫描,看看哪个类上使用到了 @Repositoy 这些注解,扫描到的就需要特殊处理将其注册到 Spring 容器。想一下,这里 Spring 其实就会对这个标签进行解析,核心代码:


registerBeanDefinitionParser("component-scan", new ComponentScanBeanDefinitionParser());

用户头像

极客good

关注

还未添加个人签名 2021.03.18 加入

还未添加个人简介

评论

发布
暂无评论
9000字通俗易懂的讲解下Java注解,你还看不明白?,linux操作系统基础与应用答案