Spring常见异常说明

文章要点

  • Spring bean 的声明方式
  • Spring bean 的注入规则
  • Spring bean 的依赖查找规则
  • Spring bean 的名称定义方式和默认名称规则

XXX required a bean of type ‘XXX’ that could not be found.

这种情况,一般是因为没有在 Spring bean 容器中找到对应的 bean 。

Spring常见异常说明

bean 的声明

使用一个 bean,首先要声明一个 bean,使之交给 Spring 管理 声明一个 bean , 常用有如下几种方式

  • xml 声明
  • 注解声明

xml 声明

这种适用于非 SpringBoot 框架,老旧 Spring MVC 的框架使用形式。

现有架构常用于 dubbo bean 的声明,所有的 dubbo 引用和服务,都会声明成为一个 Spring bean


当你发现在 Spring bean 容器中无法找到一个 dubbo 服务的时候(这时候是引用 dubbo 服务),请从如下角度去思考问题:

  • dubbo 服务的注册/引用了吗 ?
  • dubbo 服务的配置属性 interface 写的对吗 ? 接口包打了吗?
  • dubbo 服务的引用配置属性 id 和 使用 dubbo 服务的地方匹配吗?

上述问题都解决后,就不会出现找不到 dubbo bean 的问题了

注解声明

声明成 Spring bean 有一个基础的注解 @Component ,这些都用与类上。


@Component  //这个类实例 已经交给 Spring 管理了
public class ComponentService {

}

其余我们常用的 @Service@Repository 等均继承这个基础注解,都可以声明一个类的实例为 Spring bean

其继承关系如下

Spring常见异常说明

注解声明-高级版

高级版注解声明需要依赖 @Configuration 注解

上章节中的 ComponentService 也可以用如下的方式去声明


@Configuration
public class TestConfiguration {

    //声明 ComponentService
    @Bean
    public ComponentService componentService() {
        //返回一个自定义实例,交给 Spring bean 管理
        return new ComponentService();
    }
}

这种返回一个自定义 bean 的类型,常用于项目组件配置,如 DataSourceJedisPool 等,业务系统一般用不到。

bean 的注入

我们可以使用 @Autowired (Spring)、 @Resource(JSR250规范) 、 @Inject(JSR330规范) 来进行注入

通常的,我们使用 @Autowired 足以满足各种注入场景,这里不再介绍另外两种。

在一个被 Spring 管理的类实例 (画重点) 当中 引用另外一个 Spring bean 的方式,即 bean 的注入方式,有如下三种:

  • 构造函数注入(官方推荐)
  • 字段注入
  • 方法注入

首先声明一个 bean,用于被别的 bean 注入

@Repository
public class TestRepository {
}

构造函数注入

@Service
public class TestService {

    private TestRepository testRepository;

    //@Autowired
    public TestService(TestRepository testRepository) {
        this.testRepository = testRepository;
    }
}

注意在构造函数上面, @Autowired 已经被我注释掉,原因如下:

As of Spring Framework 4.3,
an @Autowired annotation on such a constructor is no longer necessary
if the target bean defines only one constructor to begin with.

字段注入

@Service
public class TestService {
    @Autowired
    private TestRepository testRepository;
}

方法注入

@Service
public class TestService {

    private TestRepository testRepository;

    @Autowired
    public void setTestRepository(TestRepository testRepository) {
        this.testRepository = testRepository;
    }
}

以上方式可以混用,不拘泥于特定的形式。推荐构造函数注入而不是字段注入,是因为

  1. 字段注入完全依赖 Spring 的 IOC 功能,耦合性较高
  2. 防止不当使用时候,Spring bean 字段为空,这个是重点
  3. 当以也应该以 final 类型修饰字段时候,避免空指针的同时,也保证了依赖的不可变,不可变即安全
  4. 保证 Spring bean 的调用方调用的时候 bean 是完全初始化状态

至此,Spring bean 的注入已经 初入门径了。

Field xxx in xxx.xxx.xxx.xx required a single bean, but 2 were found

@Autowired 是根据接口进行注入,上述问题是因为在 依赖查找 中发现同一 Interface 的多个实例 bean

场景还原如下

  1. 声明一个接口类
public interface MultiService {
}
  1. 添加两个实现类
@Service
public class MultiServiceImpl implements MultiService{
}

@Service
public class MMultiService implements MultiService{
}
  1. 注入 MultiService
@Service
public class TestService {

    //这里进行注入,然而 Spring 发现了2个实例,默认规则没有匹配上,不知道该注入哪个了
    @Autowired
    private MultiService multiService;
}

那么,在一个接口有多个实现 bean 的情况下,如何注入自己想要的 bean ?

依赖查找的部分规则

  1. 首先根据类型查找(byType),如果查找结果唯一,就返回给依赖赋值。
  2. 如果依赖不唯一,则根据名称查找(byName),bean 名称不会重复。

byName的匹配规则:

  1. 如果有 @Qualifier 指定 依赖 bean 的名称,则拿指定名称与上述查找到的 bean 的名称进行匹配,匹配到则返回给依赖赋值
  2. 如果匹配不到,则拿 @Autowired 修饰的字段名或者方法参数名与 上述查找到的 bean 的名称匹配,匹配到则返回给依赖赋值
  3. 匹配不到则报 Field xxx in xxx.xxx.xxx.xx required a single bean, but 2 were found 错误

bean 名称的指定

在声明一个 bean 的时候,可以指定 bean 名称

//这里指定了bean name 为 componentService
@Component("componentService")
public class ComponentService {
}

//这里指定了bean name 为 multiService
@Service("multiService")
public class MultiServiceImpl implements MultiService{
}

//这里指定了bean name 为 mMultiService
@Service("mMultiService")
public class MMultiService implements MultiService{
}

如果没有指定 bean 名称,默认会以当前类名的首字母小写为 bean 名称

但是如果当前类的第二个字母仍然为大写,那就会以当前类名 为 bean 名称

MMultiService 的默认 bean 名称为 MMultiService
MultiServiceImpl 的默认 bean 名称为 multiServiceImpl

在使用一个 bean 的时候,也可以指定 bean 名称

@Service
public class TestService {

    @Autowired
    @Qualifier("MMultiService")  //指定 bean 名称
    private MultiService multiService;
}

org.springframework.context.annotation.ConflictingBeanDefinitionException: Annotation-specified bean name ‘xxx’ for bean class [xxx.xx.xx.xx] conflicts with existing, non-compatible bean definition of same name and class [xxx.xxx.xxx]

这种就是 bean 名称定义重复了,知道异常栈中提示的错误 bean 名称并修改即可

The bean ‘xxx’, defined in class path resource [xx/xx/xx.class], could not be registered. A bean with that name has already been defined in file [xx\xx\xx.class] and overriding is disabled.

这种是 bean 被声明了多次,检查是否配置类编写错误,一般业务 bean 不会声明多次

这里可能会提示你通过设置 spring.main.allow-bean-definition-overriding=true 允许 bean 定义被覆盖

Action:
Consider renaming one of the beans or enabling overriding by setting spring.main.allow-bean-definition-overriding=true

请不要这么做!请找到 bean 声明冲突的地方,检查 bean 是否有特殊设置,根据业务进行取舍。不要简单覆盖 bean 定义了事

Original: https://www.cnblogs.com/qnlcy/p/15012134.html
Author: 去哪里吃鱼
Title: Spring常见异常说明

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/577912/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • Java类初始化顺序小结

    第一种情况(单一类) 测试结果 静态变量 静态&…

    Linux 2023年6月7日
    095
  • Django_渲染详解

    Django_render 模板语法 模板引擎是一种可以让开发者把服务端数据填充到html网页中完成渲染效果的技术。它实现了把前端代码和服务端代码分离的作用,让项目中的业务逻辑代码…

    Linux 2023年6月7日
    0119
  • mit 6.824 lab2A ,raft 领导人选举实现(lab2D中有关于此处大量代码修改,找出了很多错误)

    lab2 说明: https://pdos.csail.mit.edu/6.824/labs/lab-raft.html 参考博客: https://zhuanlan.zhihu….

    Linux 2023年6月7日
    0124
  • 普通 Docker 与 Kubernetes 对比

    Docker提供基本容器管理 API 和容器镜像文件格式Kubernetes 管理运行容器的(物理或虚拟)主机群集,如果 Docker 是 OCP 的”内核&#8221…

    Linux 2023年6月6日
    087
  • python 多线程

    python 多线程 多线程流程 导入模块 import threading 通过线程类型创建线程对象 线程对&a…

    Linux 2023年6月13日
    075
  • flask 之 请求钩子

    请求钩子 什么是请求钩子? 在客户端和服务器交互的过程中,有些准备工作或扫尾工作需要统一处理,为了让每个视图函数避免编写重复功能的代码, flask提供了统一的接口可以添加这些处理…

    Linux 2023年6月8日
    096
  • Python schedule 库定时任务

    Python schedule 库定时任务 schedule的使用 用于scrapy定时任务设置 import schedule import time def job(): pr…

    Linux 2023年6月13日
    092
  • Vue3

    setup 函数时,它将接受两个参数:(props、context(包含attrs、slots、emit)) setup函数是处于 生命周期函数 beforeCreate 和 Cr…

    Linux 2023年6月13日
    081
  • [转帖]shell学习之shell基础知识了解

    404. 抱歉,您访问的资源不存在。 可能是网址有误,或者对应的内容被删除,或者处于私有状态。 代码改变世界,联系邮箱 contact@cnblogs.com 园子的商业化努力-困…

    Linux 2023年5月28日
    095
  • ASP.NET Core 2.2 : 二十六. 应用JWT进行用户认证及Token的刷新

    本文将通过实际的例子来演示如何在ASP.NET Core中应用JWT进行用户认证以及Token的刷新方案。(ASP.NET Core 系列目录) 一、什么是JWT? JWT(jso…

    Linux 2023年6月7日
    0109
  • 在cmd中使用doskey来实现alias别名功能

    作为一枚网络工程师,经常就是面对一堆黑框框,也是就是终端。不同操作系统、不同厂家的目录,功能相同但是键入的命令又大不相同,这些差异化容易让脑子混乱。比如华为、思科、H3C、锐捷的设…

    Linux 2023年6月6日
    078
  • 详解Redis基本命令

    当redis环境搭建结束后,接下来需要掌握并了解redis的一些相关命令,本篇文章主要从实际操作的层面来与大家分享redis基本命令, 具体包括:Redis五大基本类型命令(Str…

    Linux 2023年5月28日
    0116
  • pod(一):Kubernetes(k8s)创建pod的两种方式

    服务器版本 docker软件版本 CPU架构 CentOS Linux release 7.4.1708 (Core) Docker version 20.10.12 x86_64…

    Linux 2023年6月7日
    079
  • 函数调用栈

    博客网址:www.shicoder.top微信:18223081347欢迎加群聊天 :452380935 这个分栏我们开始学习PWN,当然PWN也是自己的兴趣爱好,所以可能博客更新…

    Linux 2023年6月13日
    0112
  • Linux虚拟机上按安装jdk1.8.0

    Linux虚拟机上按安装jdk1.8.0 1.准备工作 jdk1.8.0下载地址: http://www.oracle.com/technetwork/java/javase/do…

    Linux 2023年6月11日
    077
  • 全域安全:一种运行时安全管理模型

    前言: 全域安全是一种新的安全管理模型,现在用在Laxcus分布式操作系统上,如果能够在ICT领域全面推广,当下计算机的安全问题,包括西工大的泄密事件,都可以避免了。以下是相关介绍…

    Linux 2023年6月6日
    092
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球