浅谈Spring注解

Spring目前的趋势是使用注解结合Java代码而不是配置来定义行为、属性、功能、规则和扩展点,因此梳理注解也是梳理Spring功能点的很好的方式,全面的梳理可以补足我们知识点的漏洞。

查找所有注解

首先,我们来创建一个项目,使用SPRING INITIALIZR生成一个引入Spring各种组件的项目模板,然后引入如下工具包:

<dependency><groupId>org.reflections</groupId><artifactId>reflections</artifactId><version>0.9.11</version>
</dependency>

通过这个反射工具包,我们可以创建一个Spring Boot应用程序,以一行代码打印出所有Spring框架的注解:

import org.reflections.Reflections;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;import java.lang.annotation.Annotation;@Component
public class ScanAnnotationRunner implements CommandLineRunner {@Overridepublic void run(String... args) throws Exception {new Reflections("org.springframework").getSubTypesOf(Annotation.class).stream().map(clazz->clazz.getName()).sorted().forEach(System.out::println);}
}

输出结果这里就不给出了,下面我们逐一进行梳理其中的一些重要注解。

有关注解

Java的Annotation注解(类似于C#的Attribute特性),说白了就是给代码打上标签的能力。我们可以配置这个标签的保留阶段,仅源代码,源代码+字节码,源代码+字节码+运行时。通过引入注解,我们可以简单快速赋予代码生命力,大大提高代码可读性和扩展性。注解本身不具有任何能力,只是一个标签,但是我们可以定义各种标签然后实现各种标签处理器来对类、方法、属性甚至参数等进行功能扩展、功能开启、属性定义、行为定义、规则定义、关联处理、元数据定义等等。在实现各种框架的时候,我们经常会自定义标签方便框架使用者仅仅通过在合适的地方引入合适的注解来启用(或自定义)框架的一些能力并应用到我们的程序中。

不仅仅是框架的作者会大量使用注解,在之前的系列文章中我们也多次自定义注解,我们有通过定义@Metrics注解配合Spring AOP来为程序启动打点、日志、异常等功能,我们有通过定义@Sign注解配合Spring MVC的ResponseBodyAdvice进行数据签名功能,我们还经常会定义各种自定义注解配合Spring MVC的HandlerMethodArgumentResolver进行权限的校验等等功能。采用这种模式,我们的核心业务逻辑可以保持清晰干净,通过注解配合AOP赋予代码额外的能力。

你可能会说,注解还是有侵入性,我们需要耦合框架定义的那些注解,这个问题其实是无解的,100%无侵入性也代表了可读性的降低,代码的功能和能力应当聚合在一起,这也就是为什么Spring现在也不建议采用XML来做配置。Java核心类库并没有什么注解,好在Spring已经有了大量注解,而Spring也变为了Java开发的标准,所以其实我们很多时候如果希望自己的框架(RPC啥的)完全没有侵入性的话可以借用Spring的那些注解@Autowired、@Controller、@Service等注解,配合各种包的规范其实我们可以对目标元素的功能识别个八九不离十,完全有可能实现0侵入的功能增强。

有关如何实现自定义注解不赘述,这里我们简单回顾一下几个元注解(注解的注解):

- @Documented:将会在被此注解注解的元素的javadoc文档中列出注解,一般都打上这个注解没坏处

- @Target:注解能被应用的目标元素,比如类、方法、属性、参数等等,需要仔细思考

- @Retention:仅在源码保留,还是保留到编译后的字节码,还是到运行时也去加载,超过90%的应用会在运行时去解析注解进行额外的处理,所以大部分情况我们都会设置配置为RetentionPolicy.RUNTIME

-@Inherited:如果子类没有定义注解的话,能自动从父类获取定义了继承属性的注解,比如Spring的@Service是没有继承特性的,但是@Transactional是有继承特性的,在OO继承体系中使用Spring注解的时候请特别注意这点,理所当然认为注解是能被子类继承的话可能会引起不必要的Bug,需要仔细斟酌是否开启继承

- @Repeatable:Java 8引入的特性,通过关联注解容器定义可重复注解,小小语法糖提高了代码可读性,对于元素有多个重复注解其实是很常见的事情,比如某方法可以是A角色可以访问也可以是B角色可以访问,某方法需要定时任务执行,要在A条件执行也需要在B条件执行

- @Native:是否在.h头文件中生成被标记的字段,除非原生程序需要和Java程序交互,否则很少会用到这个元注解

现在我们来从几个方面逐一温习一下Spring的那些常用的值得关注的注解。

Spring核心注解

- 首先来看一下各种stereotype:按分类定义了由Spring管理的各种组件,@Controller定义表现层组件,@Service定义业务逻辑层组件,@Repository定义数据访问层资源库组件,@Component定义其它组件(比如访问外部服务的组件),之前也说过了随着这些注解功能无区别,但是对组件进行合适的分类意义重大,不仅仅增加可读性而且方便我们通过AOP对不同类型的组件进行更多自动增强

- 再来看看IOC相关的一些注解:@Autowired自动装配不用多说了;@Required用于在setter方法标记属性值需要由Spring进行装配,对于目前版本的Spring这个注解已经废弃,现在Spring更推荐使用构造方法注入;@Qualifier用于通过给Bean定义修饰语来注入相应的Bean,和@Autowired一起使用相当于@Resource的效果,当然还有一种常见用法是嵌入其它注解用于对Bean进行区分,然后配合@Autowired一起使用,参见后面提到的Spring Cloud的@LoadBalanced注解;@Value用于注入属性配置或SpEL表达式(前者是我们常见用法,后者可以从其它对象获取值,功能更强大一点);@Lookup可以实现方法注入,如果我们的类是单例的,但是又希望Spring注入的依赖的对象是Prototype生命周期(每次new一个出来)的,这个时候可以通过此注解进行方法注入

- 然后来看一下有关事务的几个注解:@EnableTransactionManagement用于开启事务管理,使用Spring Boot如果引入Spring Data的话不需要手动开启(不过建议大家在使用事务的时候还是通过日志来验证事务管理是否生效);@Transactional大家都知道用于开启事务以及设置传播性、隔离性、回滚条件等;@TransactionalEventListener用于配置事务的回调方法,可以在事务提交前、提交后、完成后以及回滚后几个阶段接受回调事件

@Order注解可以设置Spring管理对象的加载顺序,在之前介绍AOP的文章中我们看到有的时候我们必须通过设置合理的@Order来合理安排切面的切入顺序避免一些问题,还有在一些业务场景中,我们往往会去定义一组类似于Filter的@Component,然后会从容器获得一组Bean,这个时候业务组件的运行顺序往往会比较重要,也可以通过这个方式进行排序

@AliasFor注解可以设置一组注解属性相互作为别名,对于有歧义的时候会使代码更清晰,此外还有一个用途是创建复合注解,Spring MVC的@GetMapping注解就是基于@RequestMapping这个注解创建的复合注解,我们可以很方便得通过这种方式来实现注解的继承

 

Spring上下文注解

- 首先来看一下配置相关的一些注解:@Configuration用于标注配置类,启用Java配置方式的Bean配置;@Bean用于配置一个Bean;@ComponentScan(@ComponentScans用于配置一组@ComponentScan,Java8可以直接使用重复注解特性配置多个@ComponentScan)用于扫描包方式配置Bean;@PropertySource以及@PropertySources用于导入配置文件;@Conditional用于设置关联的条件类,在合适的时候启用Bean的配置(Spring Boot自动配置根基);@Import用于导入其它配置类;@ImportResource用于导入非Java配置方式的XML配置;@Profile用于指定在合适的Profile下启用配置;@Lazy用于告知容器延迟到使用的时候实例化Bean(默认情况下容器启动的时候实例化Bean来检查所有的问题);@Description用于给Bean设置描述;@Scope用于设置Bean的生命周期;@Primary用于在定义了多个Bean的时候指定首选的Bean
- 其它一些注解包括:@EventListener用于设置回调方法监听Spring制定的以及自定义的各种事件;@EnableAspectJAutoProxy用于开启支持AspectJ的@Aspect切面配置支持,使用Spring Boot引入了AOP启动器的话不需要显式开启

Spring Web注解

Spring MVC的各种注解对应了SpringMVC各方面的功能,下面我们来了解一下:

- 首先是三个定义了Bean特殊生命周期的复合注解:@RequestScope@SessionScope@ApplicationScope。在Web应用中,我们可能需要Bean跟随请求、会话和应用程序的声明周期来进行创建,这个时候可以直接使用这三个快捷的复合注解

- 接下去可以看到各种@XXXMapping的注解,分别用于配置HandlerMethod匹配到不同的Http Method,当然不使用这些快捷的注解也是可以的,直接使用@RequestMapping然后手动设置method

@ResponseStatus可以用到方法上也可以用到异常上,前者会直接使请求得到指定的响应代码或原因(可以配合@ExceptionHandler使用),后者可以实现遇到指定异常的时候给出指定的响应代码或原因,@ResponseBody我们实现Restful接口的时候(@RestController)最常用了,把返回内容(序列化后)输出到请求体

- Spring MVC给了我们各种注解方便我们从HTTP请求各种地方获取参数,@RequestBody从请求体(处理复杂数据,比如JSON),@RequestHeader从请求头,@CookieValue从cookie中,@SessionAttribute从会话中,@RequestAttribute从请求的Attribute中(比如过滤器和拦截器手动设置的一些临时数据),@RequestParam从请求参数(处理简单数据,键值对),@PathVariable从路径片段,@MatrixAttribute矩阵变量允许我们采用特殊的规则在URL路径后加参数(分号区分不同参数,逗号为参数增加多个值)

@ControllerAdvice是一个重要注解,允许我们在集中的地方配置控制器(有@RequestMapping的方法)相关的增强(@RestControllerAdvice也是差不多的,只是相当于为@ExceptionHandler加上了@ResponseBody)。那么可以应用哪些增强呢?首先是可以用@ExceptionHandler进行统一的全局异常处理;第二是@InitBinder用来设置WebDataBinder,WebDataBinder用来自动绑定前台请求参数到Model中;第三是@ModelAttribute让全局的@RequestMapping都能获得在此处设置的键值对。当然,这里说的@InitBinder和@ExceptionHandler也可以不定义在@ControllerAdvice内部(作为全局开启),定义在Controller内部应用到某个Controller也是可以的

- 其它还有一些注解比如:@CrossOrigin可以用到Controller或Method上(需要配合@RequestMapping)设置细粒度的跨域行为

在之前的文章中我们也提到,对于Spring MVC,定义自己的注解应用到参数、方法、控制器上,配合HandlerMethodArgumentResolver、XXAdvise、以及Interceptor实现具体的功能来使用太太常见了,几乎所有的非业务横切关注点,我们都不应该在方法实现中重复任何一行代码。

 

Spring Boot注解

 

- 来看一下上下文相关的注解:@ConfigurationProperties很常用(配合@EnableConfigurationProperties注解来设置需要启用的配置类),用来自定义配置类和配置文件进行关联;@DeprecatedConfigurationProperty用于标记废弃的配置以及设置替代配置和告知废弃原因;@ConfigurationPropertiesBinding用于指定自定义的转换器用于配置解析的时的类型转换;@NestedConfigurationProperty用于关联外部的类型作为嵌套配置类

 

- 再看看自动配置相关的注解,自动配置是Spring Boot最重要的特性,在之前的系列文章中我有提到一个观点,IOC是好事情,但是把组件内部的一些默认配置以及组件和组件的组装交给外部用户来配置其实是不合理的,组件应当可以自动进行自我配置实现开箱急用,只有需要自定义组件的时候才要求外部来进行个性化配置:@EnableAutoConfiguration注解可以启用自动配置,Spring Boot应用程序一般我们会直接使用复合注解@SpringBootApplication@AutoConfigureOrder(值越小优先级越高)、@AutoConfigureAfter@AutoConfigureBefore用于设置自动配置类加载顺序,以及精确控制加载依赖关系,有的时候我们的自动配置需要相互依赖或者会相互干扰,需要手动调节

 

- 最后来看一下十几种配置条件,用好这些注解是实现完善的自动配置的关键:@ConditionalOnBean用于仅当容器中已经包含指定的Bean类型或名称时才匹配条件;@ConditionalOnClass仅当classpath上存在指定类时条件匹配;@ConditionalOnCloudPlatform仅当指定的云平台处于活动状态时条件匹配;@ConditionalOnExpression依赖于SpEL表达式的值的条件元素的配置注解;@ConditionalOnJava基于应用运行的JVM版本的条件匹配;@ConditionalOnJndi基于JNDI可用和可以查找指定位置的条件匹配;@ConditionalOnMissingBean仅当容器中不包含指定的Bean类型或名称时条件匹配;@ConditionalOnMissingClass仅当classpath上不存在指定类时条件匹配;@ConditionalOnNotWebApplication仅当不是WebApplicationContext(非Web项目)时条件匹配,对应@ConditionalOnWebApplication;@ConditionalOnProperty是检查指定的属性是否具有指定的值;@ConditionalOnResource表示仅当 classpath 上存在指定资源时条件匹配;@ConditionalOnSingleCandidate仅当容器中包含指定的Bean类并且可以判断只有单个候选者时条件匹配。其实所有这些实现原理都是扩展SpringBootCondition抽象类(实现之前提到的Condition接口),我们完全可以实现自己的条件注解(配合@Conditional注解关联到自己实现的SpringBootCondition)

Spring Cloud注解

在介绍本系列文章的第一篇中我们就提到了,Spring Cloud整齐划一通过各种EnableXXX注解开启某个功能,这里就不对这些注解进行说明了,使用Spring Boot组件的功能非常简单,基本就是引POM+EnableXXX+设置配置文件三部曲。

- 首先是 Netflix包下的一些注解,各种EnableXXX就不说了,参考前一篇文章,之前没介绍过@RibbonClient,这个注解用来为负载均衡客户端做一些自定义的配置,可以进一步配置或自定义从哪里获取服务端列表、负载均衡策略、Ping也就是服务鉴活策略等等

- client包下的@SpringCloudApplication之前文章中我们也没有使用到,这是一个复合注解就是@SpringBootApplication+@EnableDiscoveryClient+@EnableCircuitBreaker,Spring Cloud那堆东西很多,还是自己亲手定义一个一个功能的注解来的踏实;@LoadBalanced注解用于和RestTemplate配合使用构成一个负载均衡的Http客户端,实现原理上其实这个注解是一个@Qualifier注解,Spring会为所有@LoadBalanced的RestTemplate加入一个LoadBalancerInterceptor(实现ClientHttpRequestInterceptor)实现负载均衡

- sleuth包下面的注解和链路跟踪相关,比较常用的是通过@SpanName手动设置span的名称,其它注解对于业务开发并不常用

为了让学习变得轻松、高效,今天给大家免费分享一套Java入门教学资源。帮助大家在成为Java架构师的道路上披荆斩棘。需要资料的欢迎加入学习交流群:9285,05736

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/503267.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

linux nginx 状态,Nginx开启运行状态(status)功能 | IT运维网

1、查验Nginx的Status模块,此模块一般安装时都会安装的。2、编辑Nginx主配置文件(/usr/local/nginx/conf/nginx.conf)在默认站点中配置Status功能location /nginx_status #开启Nginx站点运行状态功能并定义状态查看访问位置{stub_status on; #启用状态access_log…

Java 并发编程必须知道的七种锁类型以及应用

锁是解决并发冲突的重要工具。在开发中我们会用到很多类型的锁&#xff0c;每种锁都有其自身的特点和适用范围。 需要深刻理解锁的理念和区别&#xff0c;才能正确、合理地使用锁。 常用锁类型 乐观锁与悲观锁 悲观锁对并发冲突持悲观态度&#xff0c;先取锁后访问数据&…

linux kvm dhcp配置,《转》QEMU-KVM创建虚拟机自动指定IP的配置

在使用qemu创建虚拟机的过程中是无法指定IP地址的&#xff0c;可是在实际应用中&#xff0c;我们是需要虚拟机拥有IP地址的&#xff0c;并且不是人为去虚拟机操作系统上配置。在qemu虚拟机技术文档(http://qemu.weilnetz.de/qemu-doc.html#pcsys_005fmonitor)里捣鼓了好久&…

谈一谈Java编程开发中的并发控制

并发指在宏观上的同一时间内同时执行多个任务。为了满足这一需求&#xff0c;现代的操作系统都抽象出 线程 的概念&#xff0c;供上层应用使用。 这篇博文不打算详细展开分析&#xff0c;而是对java并发中的概念和工具做一个梳理。 沿着并发模型、并发要解决的问题、基本工具、…

c语言中头文件及其作用,C语言头文件的作用是什么

C语言头文件的作用&#xff1a;1、头文件是程序各部分之间保证信息一致性的桥梁&#xff0c;是连接程序对象定义和使用的纽带&#xff1b;2、用于指定模块接口的声明放在文件中&#xff0c;文件名中应标明其预期用途。本文操作环境&#xff1a;Windows7系统&#xff0c;宏基S40…

Java技术大咖为什么都有写博客的习惯呢?

把自己的设计、思路、总结都写到日记里&#xff08;我用evernote&#xff09;&#xff0c;便于自己思路的整理&#xff0c;很多时候人脑的缓存是真不够用&#xff0c;后面回来找思路的时候&#xff0c;细节也都在笔记里面。 1.让我思维更清晰&#xff0c;表述更有条理 我生活中…

Java与C++有何区别呢?请看以下几点就明白了……

Java和C都是面向对象语言。也就是说&#xff0c;它们都能够实现面向对象思想。那两者到底有何区别&#xff1f;由于c为了照顾大量的C语言使用者&#xff0c; 而兼容了C&#xff0c;使得自身仅仅成为了带类的C语言&#xff0c;多多少少影响了其面向对象的彻底性&#xff01;JAVA…

零基础学习java必须要了解的学习路线

Java开发目前热火朝天&#xff0c;但是有传言说Java将被取代&#xff0c;当然那只是传言。今天小编来谈谈零基础学习Java必须了解的学习路线。 第一阶段&#xff1a;HTMLCSS 静态布局开始 有人肯能会说我是做后端的&#xff0c;前端的事情不用管&#xff0c;这本身就是一个误区…

Java开发以及Web 和移动程序员必须了解的10个框架

新的一年已经开始&#xff0c;不知道大家有没有定好小目标。如果2019年还没有决定学什么&#xff0c;那么你来对地方了。在今天的文章中&#xff0c;我将分享一些你可以学习的最好框架&#xff0c;以提升你在移动和Web开发以及大数据技术方面的知识。 在当今世界&#xff0c;对…

Java开发学习必须了解的基础知识点

面向对象和面向过程的区别 面向过程&#xff1a; 优点&#xff1a;性能比面向对象高&#xff0c;因为类调用时需要实例化&#xff0c;开销比较大&#xff0c;比较消耗资源;比如单片机、嵌入式开发、Linux/Unix等一般采用面向过程开发&#xff0c;性能是最重要的因素。 缺点&a…

Java的数据类型及其封装器类

Java基本类型共有8种&#xff0c;基本类型可以分为3类&#xff0c;字符类型char&#xff0c;布尔类型boolean和数值类型byte、short、int、long、float、double。数值类型又可以分为整数类型byte、short、int、long和浮点数类型float、double。JAVA中的数值类型不存在无符号的&…

Java开发中数据类型之间的转换

1).简单类型数据间的转换,有两种方式:自动转换和强迫转换,通常产生在表达式中或方法的参数传递时。 自动转换 具体地讲,当1个较"小"数据与1个较"大"的数据1起运算时,系统将自动将"小"数据转换成"大"数据,再进行运算。而在方法调用时…

单片机c语言怎样添加自定义头文件,单片机C语言编程与或|头文件常见问题

一、常见问题1、头文件reg51.h和reg52.h其实是一样的&#xff0c;大家两个都可以用。2、main()前面的void可加可不加&#xff0c;反正都是无返回值函数。3、不是每一个程序都要用到死循环while(1)&#xff0c;例如点亮一颗LED小灯就不用&#xff0c;只要执行一次就一直是高电平…

为什么码农要了解业务呢?网友:不是敲代码就好了吗?

分析领域的一位资深人士对我非常无助地摇了摇头。 “这个程序员&#xff0c;不要吃烟花&#xff01;”我也深深感到世界各地的农民代码都是一样的。 这让我想起了它&#xff0c;也是他。很多年前我提醒过我。——关注业务。从那以后&#xff0c;我一直在匆匆走上技术商业的双重…

论程序员如何规划职业路线?网友:从码农到工程师?

很多人可以说对这个话题两个字&#xff0c;三年或五年以上工作经验的编程&#xff0c;老建筑师&#xff0c;设计师&#xff0c;技术副总裁&#xff0c;首席技术官&#xff0c;和自己的观点和实践经验。没有一种方法适合每个人。这套针对软件工程师的专业开发计划只是您在旅途中…

c语言实现socket转json,C++实现json形式的Socket传输图片

本文实例为大家分享了C实现json形式的Socket传输图片的具体代码&#xff0c;供大家参考&#xff0c;具体内容如下大致流程&#xff1a;客户端读取图片&#xff0c;经过Base64编码&#xff0c;转成字符串的形式&#xff0c;保存到json中&#xff0c;通过socket传到服务端&#x…

经常被问到的十个 Java 面试题?你Get了吗?

1. 以满分十分来评估自己——你有多擅长 Java&#xff1f; 如果你并不完全确信你自己或是你对 Java 的熟练程度&#xff0c;那么这会是一个非常棘手的问题。如果有这种情况&#xff0c;你应该把打分调低一点。之后&#xff0c;你大概会得到与你承认的水平相符的问题。因此&…

Java虚拟机JVM的内部体系结构

JVM(Java虚拟机)是一个抽象机器。 它是一个提供可以执行Java字节码的运行时环境的规范。JVM可用于许多硬件和软件平台(即JVM是平台相关的)。 什么是JVM&#xff1f; JVM(Java虚拟机)是&#xff1a; 指定Java虚拟机的工作的规范。 但实现提供程序是独立的选择算法。 其实现是由…

Java并发编程——volatile

1. 并发编程的两个关键问题 并发是让多个线程同时履行&#xff0c;若线程之间是独立的&#xff0c;那并发实现起来很简单&#xff0c;各自履行各自的就行&#xff1b;但常常多条线程之间需要同享数据&#xff0c;此时在并发编程进程中就不可避免要斟酌两个问题&#xff1a;通讯…

Java历经20年沧桑,将持续革新

对于企业界来说&#xff0c;很多服务器都部署着Java应用程序&#xff0c;许多物联网设备也都是基于Java技术开发。 20年前&#xff0c;Java为人们带来前所未有的科技理念&#xff1a;一次编写&#xff0c;到处运行。 在此之前&#xff0c;编程语言的种类寥寥无几&#xff0c;…