【Spring笔记05】Spring的自动装配

news2024/11/18 3:39:11

这篇文章,主要介绍的内容是Spring的自动装配、五种自动装配的方式。

目录

一、自动装配

1.1、什么是自动装配

1.2、五种自动装配方式

(1)no

(2)default

(3)byType

(4)byName

(5)constructor


一、自动装配

1.1、什么是自动装配

自动装配,是指:Spring自动的建立Bean对象和Bean对象之间的依赖关系,而不需要我们开发人员手动的进行设置,这个过程就叫做自动装配。

自动装配,其实就是和之前介绍的依赖注入是一样的,只不过我们进行依赖注入的时候,需要开发人员在XML配置文件里面,通过【<property>】标签或者【<constructor-arg>】标签,手动的进行属性赋值,可以想到,如果项目中有很多对象都需要赋值,那这样就需要编写许多的XML配置代码,这不利于XML配置文件的维护。为了能够简化一下依赖注入,所以就提出了自动装配的概念,目的就是让Spring自动的替我们给属性赋值,从而简化XML配置。

Spring中默认情况下,是没有开启自动装配的功能,如果我们需要使用自动装配,则需要通过在【<bean>】标签中添加【autowire】属性,然后设置采用哪种自动装配的方式,Spring提供了五种自动装配的方式,分别是:

  • no:表示不自动装配。
  • default:表示按照【<beans>】根标签上面配置的【default-autowire】方式进行自动装配。
  • byType:表示根据Bean的数据类型进行自动装配。(如果存在多个相同数据类型的Bean,则自动装配失败,抛出异常。)
  • byName:表示根据Bean的名称进行自动装配。(根据bean的id属性值进行属性,如果存在多个相同名称的id属性值,则抛出异常。)
  • constructor:表示通过构造方法进行自动装配。(根据Bean的构造方法进行自动装配,根据构造方法参数的数据类型进行属性赋值。)

下面通过一些案例,详细的介绍一下五种自动装配的内容。

1.2、五种自动装配方式

自动装配,我们需要在XML配置文件中对应的【<bean>】标签上面,通过设置【autowire】属性,来控制采用哪种方式进行自动装配。

(1)no

当我们设置【autowire="no"】的时候,此时表示Spring不会自动装配,Bean对象之间的依赖关系需要我们自己手动进行依赖注入,【autowire】默认的属性值就是no,下面通过一个案例来看看自动装配。

创建【UserDao】测试类

public class UserDao {

    public void create() {
        System.out.println("调用UserDao类的create()方法......");
    }

}

创建【UserService】测试类

public class UserService {

    private UserDao userDao;

    public void test() {
        System.out.println("开始执行UserService类的test()方法......");
        // 这里调用 userDao 类中的 create() 方法
        userDao.create();
        System.out.println("执行结束");
    }

    public UserDao getUserDao() {
        return userDao;
    }

    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }
}

XML配置上面两个类

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- 配置 UserDao -->
    <bean id="userDao" class="com.spring.demo.pojo.UserDao"/>

    <!-- 配置 UserService -->
    <bean id="userService" class="com.spring.demo.pojo.UserService" autowire="no"></bean>

</beans>

编写【Test】测试程序

public class Test {
    public static void main(String[] args) {
        // 1、获取 ApplicationContext 容器
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
        // 2、获取 Bean 对象
        UserService userService = context.getBean("userService", UserService.class);
        // 调用方法
        userService.test();
        // 关闭容器: 这里通过关闭容器来模拟Bean被销毁的操作
        context.close();
    }
}

我们运行上面的测试程序,查看控制台输出。


运行结果会提示空指针异常,因为我们设置了【UserService】类【autowire="no"】表示不会自动装配当前类中的任何属性,所以我们在使用【UserDao】对象的时候,就会出现空指针异常。

(2)default

当我们设置【autowire="default"】时候,表示当前bean按照【<beans>】标签中【default-autowre】属性进行自动装配。


我们按照上面的配置方式,再次运行测试程序,此时就可以运行成功。


因为Spring会根据【byName】进行自动装配,在UserService类中存在userDao的Bean对象,Spring就会去查找当前IOC容器里面是否存在一个叫做【userDao】的对象,如果找到则将其赋值到UserService类里面的userDao属性,此时自动装配成功。

如果Spring没要找到userDao,或者找到多个userDao对象,那么就会抛出异常,自动装配失败。

(3)byType

byType表示根据数据类型进行自动装配,只需要设置【autowire="byType"】即可。下面看个测试案例,来看看byType自动装配的效果。

创建【Dao】接口

public interface Dao {
    void create();
}

创建【UserDao1】测试类

public class UserDao1 implements Dao {
    public void create() {
        System.out.println("调用UserDao1类的create()方法......");
    }
}

创建【UserDao2】测试类

public class UserDao2 implements Dao {
    public void create() {
        System.out.println("调用UserDao2类的create()方法......");
    }
}

创建【UserService2】测试类

public class UserService2 {

    private Dao dao;

    public void test() {
        System.out.println("开始执行UserService类的test()方法......");
        // 这里调用 dao 类中的 create() 方法
        dao.create();
        System.out.println("执行结束");
    }

    public Dao getDao() {
        return dao;
    }

    public void setDao(Dao dao) {
        this.dao = dao;
    }
}


XML配置上面的测试类对象

这里我们先配置一个【UserDao1】和【UserServce2】,然后运行测试程序查看结果。


运行测试程序,控制台可以正常输出。


但是,当我们Spring中存在多个相同数据类型的Bean对象时候,此时采用byType进行自动装配就会失败。下面我们将【UserDao2】类也配置到XML里面。


再次运行测试程序,可以发现控制台抛出异常。


上面报错信息大致意思是:预期一个匹配的Bean,但是找到了2个,分别是userDao1和userDao2。

(4)byName

byName和byType类型,只不过设置【autowire="byName"】的时候,是根据【<bean>】标签的【id】属性值进行自动装配。Spring会去查找是否存在【id】和当前类里面名称相同的属性,如果存在,则进行自动装配,如果有多个或者一个都没有,则自动装配失败。

测试类还是和前面一样,不一样的地方是XML配置。

我们将XML配置修改为如下内容。


通过上面的配置,我们就已经成功设置【autowire="byName"】自动装配了,运行测试程序,查看是否自动装配成功。


然后,我们在添加【UserDao2】的XML配置,此时配置了两个名称叫做dao的Bean对象,再次运行测试程序,就会抛出异常。


以上,就是通过byName进行自动装配,就记住:多个时候,Spring也不知道该赋值哪个,所以就会报错。 

(5)constructor

constructor自动装配是根据构造方法的参数数据类型实现的,根据Bean对象的构造方法中的参数数据类型,然后Spring框架会找到相同数据类型的Bean,通过构造方法进行赋值,和byType类似。

创建【UserDao3】测试类

public class UserDao3 {
    public void create() {
        System.out.println("调用UserDao3类的create()方法......");
    }
}

创建【UserService3】测试类

public class UserService3 {

    private UserDao3 userDao3;

    // 通过构造方法注入
    public UserService3(UserDao3 userDao3) {
        this.userDao3 = userDao3;
    }

    public void test() {
        System.out.println("开始执行UserService3类的test()方法......");
        // 这里调用 dao 类中的 create() 方法
        userDao3.create();
        System.out.println("执行结束");
    }
}

XML配置上面两个Bean对象

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- 配置 UserDao3 -->
    <bean id="userDao3" class="com.spring.demo.pojo3.UserDao3"/>

    <!-- 配置 UserService3 -->
    <bean id="userService3" class="com.spring.demo.pojo3.UserService3"
          autowire="constructor"></bean>

</beans>

编写测试程序,查看控制台输出结果。

public class Test05 {
    public static void main(String[] args) {
        // 1、获取 ApplicationContext 容器
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring05.xml");
        // 2、获取 Bean 对象
        UserService3 userService = context.getBean("userService3", UserService3.class);
        // 调用方法
        userService.test();
        // 关闭容器: 这里通过关闭容器来模拟Bean被销毁的操作
        context.close();
    }
}

运行程序,控制台正常输出。


到此,Spring的自动装配的五种方式介绍完了,实际开发中,最常用的是【byType】、【byName】、【constructor】这三个,并且在之后的注解开发里面,都不会使用XML进行自动装配,而是采用【@Autowired】注解,这个注解就是默认采用【byType】进行自动装配的。


综上,这篇文章就结束啦,主要介绍的内容是Spring的自动装配、五种自动装配的方式。

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

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

相关文章

e为底数的指数运算e^x,math.exp(x)

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 e为底数的指数运算e^x math.exp(x) 选择题 关于以下代码的说法中正确的是&#xff1f; import math print("【执行】math.exp(0)") print(math.exp(0)) print("【执行】math.ex…

网络基础知识面试题1

VC++常用功能开发汇总(专栏文章列表,欢迎订阅,持续更新...)https://blog.csdn.net/chenlycly/article/details/124272585C++软件异常排查从入门到精通系列教程(专栏文章列表,欢迎订阅,持续更新...)

js 事件参考

事件参考 事件介绍 触发事件是为了通知代码可能影响代码执行的“有趣变化”。这些可能来自用户交互&#xff0c;例如使用鼠标或调整窗口大小&#xff0c;底层环境状态的变化(例如&#xff0c;低电量或来自操作系统的媒体事件)以及其他原因。 每个事件都由一个基于Event接口的…

使用Plotly模拟远古博弈游戏_掷骰子

不乏投资大师、量化基金经理从着迷博弈游戏开始迈出步伐...... 开始学习使用python包Plotly模拟掷骰子。 安装Plotly 终端输入命令&#xff1a;python3 -m pip install --user plotly 创建骰子类 掷骰子 分析结果 绘制直方图 程序都正常运行&#xff0c;直方图也显示无误&…

老胡的周刊(第110期)

老胡的信息周刊[1]&#xff0c;记录这周我看到的有价值的信息&#xff0c;主要针对计算机领域&#xff0c;内容主题极大程度被我个人喜好主导。这个项目核心目的在于记录让自己有印象的信息做一个留存以及共享。 &#x1f3af; 项目 bitwarden[2] 一个开源&#xff0c;免费&…

代码随想录算法训练营第23期day14|二叉树层序遍历、226.翻转二叉树、101. 对称二叉树

目录 一、二叉树层序遍历 非递归法 递归法 相关题目&#xff08;10题&#xff09; 二、&#xff08;leetcode 226&#xff09;翻转二叉树 递归法 层序遍历 深度优先遍历 1&#xff09;非统一写法——前序遍历 2&#xff09; 统一写法——前序遍历 三、&#xff08;le…

解决spawn-fcgi:child exited with: 127/126/1报错

解决spawn-fcgi:child exited with: 126报错 执行文件的权限不够&#xff0c;如果是使用.sh文件进行执行的&#xff0c;首先对.sh文件进行权限修改 chmod 777 执行文件.sh 之后再对sh文件中所有执行spawn-fcgi的程序授予权限 比如&#xff1a; spawn-fcgi -a 127.0.0.1 -p 789…

【无标题】This project has been opened by another efinity instance

This project has been opened by another efinity instance 说明&#xff1a;&#xff08;1&#xff09;软件自动即出可能有些进程没有关闭 &#xff08;2&#xff09;目录中有中文路径。

对一门不是非常熟悉的语言是怎么面试的

公司是一个基础通讯类的公司&#xff0c;需要的职位是一个高级系统和软件工程师。 职位要求&#xff0c;是一个完全不怎么大众的语言&#xff1a;Elixir。 没听过&#xff0c;这就对了&#xff0c;这是一个函数式的语言&#xff0c;可以认为是 Erlang 的升级版本&#xff0c;…

15073-2014 铸造钛及钛合金 知识梳理

声明 本文是学习GB-T 15073-2014 铸造钛及钛合金.pdf而整理的学习笔记,分享出来希望更多人受益,如果存在侵权请及时联系我们 1 范围 本标准规定了铸造钛及钛合金的牌号、代号和化学成分&#xff0c;以及化学成分分析方法。 本标准适用于机加石墨型、捣实型、金属型和熔模精…

BootstrapBlazor企业级组件库:前端开发的革新之路

作为一名Web开发人员&#xff0c;开发前端我们一般都是使用JavaScript&#xff0c;而Blazor就是微软推出的基于.Net平台交互式客户Web UI 框架&#xff0c;可以使用C#替代JavaScript&#xff0c;减少我们的技术栈、降低学习前端的成本。 而采用Blazor开发&#xff0c;少不了需…

[SWPUCTF 2021 新生赛]sql - 联合注入

这题可以参考文章&#xff1a;[SWPUCTF 2021 新生赛]easy_sql - 联合注入||报错注入||sqlmap 这题相比于参考文章的题目多了waf过滤 首先&#xff0c;仍然是网站标题提示参数是wllm 1、fuzz看哪些关键字被过滤&#xff1a;空格、substr、被过滤 2、?wllm-1/**/union/**/selec…

【Java】CompletableFuture学习记录

目录 介绍创建异步对象计算完成时回调方法handle 方法线程串行化方法两任务组合 - 都要完成两任务组合 - 一个完成多任务组合 介绍 业务场景&#xff1a;查询商品详情页的逻辑比较复杂&#xff0c;有些数据还需要远程调用&#xff0c;必然需要花费更多的时间。 假如商品详情页…

socket网络编程中设置socket选项的ioctlsocket、setsockopt和WSAIoctl函数的使用(附源码)

VC常用功能开发汇总&#xff08;专栏文章列表&#xff0c;欢迎订阅&#xff0c;持续更新...&#xff09;https://blog.csdn.net/chenlycly/article/details/124272585C软件异常排查从入门到精通系列教程&#xff08;专栏文章列表&#xff0c;欢迎订阅&#xff0c;持续更新...&a…

AI发展历史

一、AI的发展历史 二、AI发展的第五阶段 &#xff08;一&#xff09;、第一阶段 1.艾伦图灵与模仿游戏 艾伦•图灵&#xff08;Alan Turing&#xff0c;1912~1954&#xff09;是英国数学家、逻辑学家&#xff0c;被称为计算机科学之父&#xff0c;人工智能之父。二战中协助军…

vue重修004【下部】

文章目录 版权声明非父子通信event bus 事件总线实现步骤代码演示 非父子通信-provide&inject语法注意代码演示 v-model原理表单类组件封装& v-model 简化代码.sync修饰符语法代码示例 ref 和 $refs语法代码演示 异步更新 & $nextTick引子$nextTick演示 版权声明 …

javaee ssm框架项目添加分页控件

搭建ssm框架项目 参考上一篇博文 添加分页控件 引入依赖 <?xml version"1.0" encoding"UTF-8"?><project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schema…

关联规则挖掘(下):数据分析 | 数据挖掘 | 十大算法之一

⭐️⭐️⭐️⭐️⭐️欢迎来到我的博客⭐️⭐️⭐️⭐️⭐️ &#x1f434;作者&#xff1a;秋无之地 &#x1f434;简介&#xff1a;CSDN爬虫、后端、大数据领域创作者。目前从事python爬虫、后端和大数据等相关工作&#xff0c;主要擅长领域有&#xff1a;爬虫、后端、大数据…

Kubernetes安装部署 1

本文主要描述kubernetes的安装部署&#xff0c;kubernetes的安装部署主要包括三个关键组件&#xff0c;其中&#xff0c;包括kubeadm、kubelet、kubectl&#xff0c;这三个组件的功能描述如下所示&#xff1a; Kubeadm 用于启动与管理kubernetes集群 Kubelet 运行在所有集群的…

[晕事]今天做了件晕事21;设置代理访问网站的时候需注意的问题

今天在家上班&#xff0c;设置好VPN&#xff0c;通过代理来访问公司内部的一个系统浏览器的反应如下&#xff1a; Hmmm… can’t reach this page ***.com refused to connect. 这个返回的错误&#xff0c;非常的具有迷惑性&#xff0c;提示的意思&#xff1a;拒绝链接&#xf…