9000字通俗易懂的讲解下Java注解,你还看不明白?

这就是一个注释,那么注释有什么用呢?简单来说就是对相关的类或者方法加以说明,比如这里的Test类,上面的注释大致告诉我们这类是谁编写的,做什么的以及何时编写的这些信息,当然,信息其实还可以有更多。

所以,你要明白,注释是干嘛的,是给我们这些程序员看的,看到注释我们就明白了,哦,这个类原来是这样的……

注释是给程序员看的,那么注解呢?相差一个字,注解是给程序看的,先记住即可。

进一步理解注解

上面我们说了,注解和注释是很像的,注释是给我们人看的,注解就是给程序看的,前面的好理解,这个注解是给程序看的,你或许还有一点懵,我进一步解释下。

首先,直观感觉下什么是注解,比如我们在代码中写的这个:

/**

  • @Description 用户类
  • @Author ithuangqing
  • @Date 2020-07-31 15:33
    **/
    @Repository
    public class UserBean {
    private String name;
    private int age;
    }

这里的@Repository就是一个注解,看这段代码,上面还有注释,我们看到注释,发现都看得明白,但是看到@Repository之后,就不那么明白,这是啥,有啥用?

于是我们查,大概知道,哦,这是个注解,有了这个注解,UserBean就会被装载进Spring容器中,我们可以知道这些信息,但是,它实际上做了哪些事情,是如何起作用,也就是如何把UserBean交给Spring去处理的,这个内部实现,我们不清楚。

但是,我们虽然不清楚,有个东西它清楚,什么呢?就是一个特定的程序,也就是说,有一个专门的程序,当它看到这个UserBean上面有一个@Repository之后,这个程序就知道了,原来要把这个UserBean装载进Spring容器中,于是这个程序员就开始执行一系列的操作区把这个UserBean装载进Spring容器中。

所以,你到此应该明白:

注释是给人看到的,注解是给程序看的。

我们再进一步总结下什么是注解:

在程序中,可以把注解看成一种特殊的标记,一般是用来标记类,方法或者接口等,这些标记有一些特点,比如可以在编译的时候,(javac命令把java源文件编译成字节码文件class),类加载和运行的时候(使用java命令执行字节码文件的时候,类的生命周期开始,从加载到卸载)被读取到(一般是有专门的程序去读区这些注解,利用反射技术去解析注解,然后根据得到的信息做相应的处理)

是的,关于注解,你要知道这么一个知识点了:

对于注解,是有专门的程序去读取它,解析它,然后根据得到的消息去执行相应的操作。

关于这个程序,要看具体的场景,也就是说这个程序是不同的,那么关于这个程序怎么知道读到哪个注解该干嘛,这个还是依靠注解本身的定义,比如@Repository注解被定义成是把被注解的装载进Spring容器中,那么特有的程序获取到这个注解就知道该干什么事了。

到此,你应该知道什么是注解了,当然,是概念上的一些东西,另外,对注解是怎么起作用的,你也应该有点内味了……

注解的简单分类

这个知识点很是轻松愉快,不需要你失去多少脑细胞。

注解是有分类的,一般有三种类型的注解:

  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. 读取并执行相应流程

下面我们就以@Repository这个注解来看看这三个流程,首先是定义注解,这个我们可以在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的基础上发而来。

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Java)

最后

小编在这里分享些我自己平时的学习资料,由于篇幅限制,pdf文档的详解资料太全面,细节内容实在太多啦,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!

程序员代码面试指南 IT名企算法与数据结构题目最优解

这是” 本程序员面试宝典!书中对IT名企代码面试各类题目的最优解进行了总结,并提供了相关代码实现。针对当前程序员面试缺乏权威题目汇总这一-痛点, 本书选取将近200道真实出现过的经典代码面试题,帮助广“大程序员的面试准备做到万无一失。 “刷”完本书后,你就是“题王”!

image.png

《TCP-IP协议组(第4版)》

本书是介绍TCP/IP协议族的经典图书的最新版本。本书自第1版出版以来,就广受读者欢迎。

本书最新版进行」护元,以体境计算机网络技不的最新发展,全书古有七大部分共30草和7个附录:第一部分介绍一些基本概念和基础底层技术:第二部分介绍网络层协议:第三部分介绍运输层协议;第四部分介绍应用层协议:第五部分介绍下一代协议,即IPv6协议:第六部分介绍网络安全问题:第七部分给出了7个附录。

image.png

Java开发手册(嵩山版)

这个不用多说了,阿里的开发手册,每次更新我都会看,这是8月初最新更新的**(嵩山版)**

image.png

MySQL 8从入门到精通

本书主要内容包括MySQL的安装与配置、数据库的创建、数据表的创建、数据类型和运算符、MySQL 函数、查询数据、数据表的操作(插入、更新与删除数据)、索引、存储过程和函数、视图、触发器、用户管理、数据备份与还原、MySQL 日志、性能优化、MySQL Repl ication、MySQL Workbench、 MySQL Utilities、 MySQL Proxy、PHP操作MySQL数据库和PDO数据库抽象类库等。最后通过3个综合案例的数据库设计,进步讲述 MySQL在实际工作中的应用。

image.png

Spring5高级编程(第5版)

本书涵盖Spring 5的所有内容,如果想要充分利用这一领先的企业级 Java应用程序开发框架的强大功能,本书是最全面的Spring参考和实用指南。

本书第5版涵盖核心的Spring及其与其他领先的Java技术(比如Hibemate JPA 2.Tls、Thymeleaf和WebSocket)的集成。本书的重点是介绍如何使用Java配置类、lambda 表达式、Spring Boot以及反应式编程。同时,将与企业级应用程序开发人员分享一些见解和实际经验,包括远程处理、事务、Web 和表示层,等等。

image.png

JAVA核心知识点+1000道 互联网Java工程师面试题

image.png

image.png

企业IT架构转型之道 阿里巴巴中台战略思想与架构实战

本书讲述了阿里巴巴的技术发展史,同时也是-部互联网技 术架构的实践与发展史。

image.png
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

JAVA核心知识点+1000道 互联网Java工程师面试题

[外链图片转存中…(img-WdPpEYQL-1713807077398)]

[外链图片转存中…(img-2Y7zicdB-1713807077398)]

企业IT架构转型之道 阿里巴巴中台战略思想与架构实战

本书讲述了阿里巴巴的技术发展史,同时也是-部互联网技 术架构的实践与发展史。

[外链图片转存中…(img-AZb94kwz-1713807077398)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值