学习Spring,@Autowired的这几个易错点你一定要知道!

简介: 前言你是否懂得@Autowired注解在使用上的细节?你是否在解决因@Autowired产生的异常而无处下手?你是否了解@Autowired的基本原理?我们在做项目的过程中,随着项目规模的增大、第三方服务的引入,项目中类与类之间的依赖关系错综复杂,而@Autowired作为在依赖注入中最常使用到的注解,若没有一定的知识储备,遇到问题就直接百度搜报错信息的话,很大程度会造成一杯茶一包烟,一个bug搞一天。本篇文章我将从以上三个问题出发讲解@Autowired使用中容易出错的地方。一.@Autowired的使用细节1.依赖注入:@Autowired它可以对类成员变量、方法及构造函

前言

  1. 你是否懂得@Autowired注解在使用上的细节?
  2. 你是否在解决因@Autowired产生的异常而无处下手?
  3. 你是否了解@Autowired的基本原理?

我们在做项目的过程中,随着项目规模的增大、第三方服务的引入,项目中类与类之间的依赖关系错综复杂,而@Autowired作为在依赖注入中最常使用到的注解,若没有一定的知识储备,遇到问题就直接百度搜报错信息的话,很大程度会造成一杯茶一包烟,一个bug搞一天

本篇文章我将从以上三个问题出发讲解@Autowired使用中容易出错的地方。

一.@Autowired的使用细节

1.依赖注入:@Autowired它可以对类成员变量方法及构造函数三处地方进行标注。标注在函数上时,@Autowired会自动识别函数上的参数,接着从Spring容器中找到对应的Bean进行依赖注入,同时也可以搭配@Qualifier解决歧义问题。

2.多个匹配的Bean的处理:@Autowired默认按照byType(属性类型)装配方式,如果遇到多个匹配的Bean或者需要根据名称进行装配,可以结合@Qualifier注解来指定要注入的Bean名称。

3.@Autowired的可选性: 使用@Autowired(required = false)可以将依赖标记为可选的。如果找不到匹配的Bean,将不会抛出异常,但需要注意处理依赖缺失的情况以防止NPE异常。

4.@Autowired标注的字段的引用:在使用了@Autowired注解的Java中类的初始化顺序为

静态变量->静态初始化块->变量初始化->初始化块->构造器->@Autowired标注的变量赋值

也就是说标注了@Autowired注解的变量要等到类完全加载完才会将相应的bean注入。所以不要在构造器中使用被@Autowired注解标注的变量。

二.@Autowired常见的棘手的异常

以下几种异常是边试我们在使用@Autowired注解中经常遇到的。

  1. BeanCreationException:Bean创建失败

这种问题可能是由于Bean的构造函数抛出异常、初始化失败等原因引起的。

  1. UnsatisfiedDependencyException:存在多个匹配的Bean
  2. 在我们日常使用Springboot开发中此情况常见于一个service接口有多个实现类,因为@Autowired注解进行依赖注入时是默认按属性类型的,此时对该service进行依赖注入时,容器中出现多个类型相同bean(因为它有多个实现类),容器不知道为它注入哪个,只能走抛异常的方式。

运行结果:

  1. NoSuchBeanDefinitionException

当Spring容器无法找到与@Autowired注解所需类型匹配的Bean时,就会抛出
NoSuchBeanDefinitionException异常。

  1. BeanCurrentlyInCreationException:循环依赖问题

使用@Autowired时,如果存在循环依赖(A依赖B,B又依赖A),会导致
BeanCurrentlyInCreationException异常。

运行结果:

5.NullPointException:空指针异常

没错,NullPointException也是使用Autowired常常碰到的异常。这种异常经常出现在依赖未注入而导致的。@Autowired的使用细节目录中提到的1,2,4都有可能导致。

三.@Autowired底层简述

在Spring中有着一个后处理器的概念,每一个后处理器都有着解析一种或者多种注解的功能。@Autowired正是由
AutowiredAnnotationBeanPostProcessor进行解析的。它的流程分为两步,第一:
找到类中@Autowired标注的的属性或者函数;第二:到容器去找到对应类型的bean去注入

这个方法具体在
AutowiredAnnotationBeanPostProcessor
的258行

接着我们做个测试,进行debug

这是用到的类,这里的UserMapper故意没标注@Mapper注解,模拟出错情况。

蓝色那一行,底层根据beanName:“userServiceImpl”到缓存中找到类的元数据,

返回的元数据中有userServiceImpl中用@Autowired标注的属性“userMapper”

接着进行一个数据校验(非空判断)

最终来到依赖注入阶段

发现容器中并没有匹配的bean(前面故意没在mapper接口上加mapper注解),执行报异常的操作

以防上图看的不清楚

作者:与驴OO

链接:
https://juejin.cn/post/7260763321480790053

来源:稀土掘金

著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

相关文章
|
1月前
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
63 9
|
2月前
|
前端开发 Java 数据库
SpringBoot学习
【10月更文挑战第7天】Spring学习
41 9
|
1月前
|
Java Kotlin 索引
学习Spring框架特性及jiar包下载
Spring 5作为最新版本,更新了JDK基线至8,修订了核心框架,增强了反射和接口功能,支持响应式编程及Kotlin语言,引入了函数式Web框架,并提升了测试功能。Spring框架可在其官网下载,包括文档、jar包和XML Schema文档,适用于Java SE和Java EE项目。
32 0
|
2月前
|
XML Java 数据格式
Spring学习
【10月更文挑战第6天】Spring学习
27 1
|
2月前
|
Java 测试技术 开发者
springboot学习四:Spring Boot profile多环境配置、devtools热部署
这篇文章主要介绍了如何在Spring Boot中进行多环境配置以及如何整合DevTools实现热部署,以提高开发效率。
100 2
|
2月前
|
前端开发 Java 程序员
springboot 学习十五:Spring Boot 优雅的集成Swagger2、Knife4j
这篇文章是关于如何在Spring Boot项目中集成Swagger2和Knife4j来生成和美化API接口文档的详细教程。
194 1
|
2月前
|
Java API Spring
springboot学习七:Spring Boot2.x 拦截器基础入门&实战项目场景实现
这篇文章是关于Spring Boot 2.x中拦截器的入门教程和实战项目场景实现的详细指南。
34 0
springboot学习七:Spring Boot2.x 拦截器基础入门&实战项目场景实现
|
2月前
|
Java API Spring
springboot学习六:Spring Boot2.x 过滤器基础入门&实战项目场景实现
这篇文章是关于Spring Boot 2.x中过滤器的基础知识和实战项目应用的教程。
36 0
springboot学习六:Spring Boot2.x 过滤器基础入门&实战项目场景实现
|
2月前
|
Java 关系型数据库 MySQL
springboot学习五:springboot整合Mybatis 连接 mysql数据库
这篇文章是关于如何使用Spring Boot整合MyBatis来连接MySQL数据库,并进行基本的增删改查操作的教程。
213 0
springboot学习五:springboot整合Mybatis 连接 mysql数据库
|
2月前
|
Java Spring
springboot 学习十一:Spring Boot 优雅的集成 Lombok
这篇文章是关于如何在Spring Boot项目中集成Lombok,以简化JavaBean的编写,避免冗余代码,并提供了相关的配置步骤和常用注解的介绍。
122 0
下一篇
DataWorks