【Spring基础】关于Spring IoC的那些事

在这里插入图片描述

文章目录

  • 一、如何理解IoC
    • 1.1 Spring IOC 概述
    • 1.2 IoC 是什么
  • 二、Ioc 配置的方式
    • 2.1 xml 配置
    • 2.2 Java 配置
    • 2.3 注解配置
  • 三、依赖注入的方式
    • 3.1 setter方式
    • 3.2 构造函数
    • 3.3 注解注入
  • 小结

一、如何理解IoC

1.1 Spring IOC 概述

  控制反转 IoC(Inversion of Control)是一种设计思想,DI (依赖注入)是实现 IoC 的一种方法,也有人认为 DI 只是 IoC 的另一种说法。没有 IoC 的程序中我们使用面向对象编程对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方。传统应用程序都是由我们在类内部主动创建依赖对象,从而导致类与类之间高耦合,难于测试,如下图所示:
在这里插入图片描述

  要理解IoC,需要弄清楚 「到底什么被反转了?如何反转的」?IoC 是指容器控制程序对象之间的关系,而不是传统实现中,由程序代码直接操控。控制权由应用代码中转到了外部容器,控制权的转移就是所谓反转。对于 Spring 而言,就是由 Spring 来控制对象的生命周期和对象之间的依赖关系。所有的类都会在 spring 容器中登记,告诉 spring 这是个什么东西,你需要什么东西,然后 spring 会在系统运行到适当的时候,把你要的东西主动给你,同时也把你交给其他需要你的东西。所有的类的创建、销毁都由 spring 来控制,也就是说控制对象生存周期的不再是引用它的对象,而是 spring。
  IoC 还有另外一个名字——“依赖注入(Dependency Injection)”。从名字上理解,所谓依赖注入,即组件之间的依赖关系由容器在运行期决定,即由容器动态地将某种依赖关系注入到组件之中。所以对象与对象之间是松散耦合,这样也方便测试,利于功能复用,更重要的是使得程序的整个体系结构变得非常灵活。

1.2 IoC 是什么

  IoC 不是什么技术,而是一种设计思想,一个重要的面向对象编程的法则。在Java开发中,Ioc意味着将你设计好的对象交给容器控制,而不是传统的在你的对象内部直接控制。我们来深入分析一下:

  • 谁控制谁,控制什么:传统 Java SE程序设计,我们直接在对象内部通过new进行创建对象,是程序主动去创建依赖对象。而 IoC 是有专门一个容器来创建这些对象,即由 Ioc 容器来控制对象的创建。关于谁控制谁?当然是 IoC 容器控制了对象;至于控制什么?那就是主要控制了外部资源获取(不只是对象包括比如文件等)。
  • 为何是反转,哪些方面反转了:有反转就有正转,传统应用程序是由我们自己在对象中主动控制去直接获取依赖对象,也就是正转;而反转则是由容器来帮忙创建及注入依赖对象。那为何是反转?因为由容器帮我们查找及注入依赖对象,对象只是被动的接受依赖对象,所以是反转。至于哪些方面反转了?依赖对象的获取被反转了。

  在传统程序设计下,都是主动去创建相关对象然后再组合起来,我们需要用到那个类的对象,只需要 new 它就可以产生该类的对象了,然在在通过该对象调用它的属性和方法。如下图所示:
在这里插入图片描述

  类和类之间都是自行产生其他类的对象来进行使用,好处是使用简单,需要的时候 new 就行。但是缺点就有很多,用到哪些类都需要自身去创建,依赖度太高,不利于解耦,也有可能多个类都使用一个类的对象,但要产生多个对象,不能“共享”,冗余过多。当有了 IoC/DI 的容器后,在客户端类中不再主动去创建这些对象了,所有对象的产生都交给了容器来实现,当需要对象时,在通过容器获取它就行。如下图所示:
在这里插入图片描述

二、Ioc 配置的方式

  众所周知,Spring设计的两个大的要点:IOC和AOP。从框架的设计角度而言,更为重要的是简化开发,比如提供更为便捷的配置Bean的方式,直至0配置(即约定大于配置)。

2.1 xml 配置

  顾名思义,就是将bean的信息配置xml文件里,通过Spring加载文件为我们创建bean。这种方式出现很多早前的SSM项目中,将第三方类库或者一些配置工具类都以这种方式进行配置,主要原因是由于第三方类不支持Spring注解。使用这种配置的优点是可以使用于任何场景,结构清晰、通俗易懂;当然其缺点也比较明显,就是配置繁琐、不易维护,枯燥无味且扩展性差。

<?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/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd"><bean id="userService" class="org.dllwh.springframework.service.UserServiceImpl"><property name="userDao" ref="userDao"/></bean>
</beans>

2.2 Java 配置

  将类的创建交给我们配置的JavcConfig类来完成,Spring只负责维护和管理,采用纯Java创建方式。其本质上,就是把在XML上的配置声明转移到Java配置类中。这种方式适用于任何场景,配置方便,因为是纯Java代码,扩展性高,十分灵活。但由于是采用Java类的方式,声明不明显,如果大量配置,可读性比较差。
  实现这种配置,需要创建一个配置类, 添加 @Configuration 注解声明为配置类。然后创建方法,在方法上加上 @bean,该方法用于创建实例并返回,该实例创建后会交给spring管理,方法名建议与实例名相同(首字母小写)。

@Configuration
public class BeansConfig {@Bean("userDao")public UserDaoImpl userDao() {return new UserDaoImpl();}@Bean("userService")public UserServiceImpl userService() {UserServiceImpl userService = new UserServiceImpl();userService.setUserDao(userDao());return userService;}
}

2.3 注解配置

  通过在类上加注解的方式,来声明一个类交给 Spring 管理,Spring 会自动扫描带有 @Component@Controller@Service@Repository 这四个注解的类,然后帮我们创建并管理,前提是需要先配置Spring的注解扫描器。使用这种方式开发便捷,通俗易懂,方便维护。当然其具有局限性,对于一些第三方资源,无法添加注解,只能采用XML或JavaConfig的方式配置。如下所示:

@Service
public class UserServiceImpl {@Autowiredprivate UserDaoImpl userDao;public List<User> findUserList() {return userDao.findUserList();}
}

一些常用的注解,包括但不限于产生对象、引入依赖、设置配置等,如下表所示:

名称解释
@Component启动 Spring 后,会自动把它转成容器管理的 Bean
@Repository用于对DAO层注解,但是目前该功能与 @Component 相同
@Service用于对业务层注解,但是目前该功能与 @Component 相同
@Controller用于对控制层注解,但是目前该功能与 @Component 相同
@Scope作用域,等同于 XML 中的 scope 写法例如: @Component(“guanwei”) @Scope(“prototype”)
@Resource默认是按照名称来装配注入的,只有当找不到与名称匹配的 bean 才会按 照类型来装配注入
@Autowired默认是按照类型装配注入的,如果想按照名称来转配注入,则需要结合 @Qualifier 一起使用
@Qualifier@Autowired 和 @Qualifier 结合使用时,自动注入的策略就从 byType 转变成 byName 了
@Configuration配置注解,标明当前类是一个配置类
@ComponentScan配置扫描那些包,一次可以扫描多个包
@Bean将方法的返回结果存放到 IoC 容器,就是手动产生对象并存放到 IoC 容器中
@Lazy设置当前类是否懒加载,配合产生对象注解使用

三、依赖注入的方式

常用的注入方式主要有三种:构造方法注入(Construct注入)、setter注入、基于注解的注入(接口注入)。

3.1 setter方式

  在XML配置方式中,property都是setter方式注入,比如下面的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/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd"><bean id="userService" class="org.dllwh.springframework.service.UserServiceImpl"><property name="userDao" ref="userDao"/></bean>
</beans>

  本质上包含两步:第一步,需要 new UserServiceImpl() 创建对象,所以需要默认构造函数;第二步,调用 setUserDao() 函数注入 userDao 的值,所以需要 setUserDao() 函数。所以对应的service类是这样的:

public class UserServiceImpl {private UserDaoImpl userDao;public UserServiceImpl() {}public List<User> findUserList() {return this.userDao.findUserList();}public void setUserDao(UserDaoImpl userDao) {this.userDao = userDao;}
}

  而在注解和Java配置方式下:

public class UserServiceImpl {private UserDaoImpl userDao;public List<User> findUserList() {return this.userDao.findUserList();}@Autowiredpublic void setUserDao(UserDaoImpl userDao) {this.userDao = userDao;}
}

  在 Spring 3.x 刚推出的时候,推荐使用注入的就是这种, 但是这种方式比较麻烦,所以在 Spring4 .x 版本中推荐构造函数注入。

3.2 构造函数

  在XML配置方式中,<constructor-arg> 是通过构造函数参数注入,比如下面的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/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd"><bean id="userService" class="tech.pdai.springframework.service.UserServiceImpl"><constructor-arg name="userDao" ref="userDao"/></bean>
</beans>

  本质上是new UserServiceImpl(userDao)创建对象,所以对应的service类是这样的:

public class UserServiceImpl {private final UserDaoImpl userDao;public UserServiceImpl(UserDaoImpl userDaoImpl) {this.userDao = userDaoImpl;}public List<User> findUserList() {return this.userDao.findUserList();}
}

  在注解和Java配置方式下

@Service
public class UserServiceImpl {private final UserDaoImpl userDao;@Autowiredpublic UserServiceImpl(final UserDaoImpl userDaoImpl) {this.userDao = userDaoImpl;}public List<User> findUserList() {return this.userDao.findUserList();}
}

3.3 注解注入

  以@Autowired(自动注入)注解注入为例,修饰符有三个属性:Constructor、byType、byName,默认按照byType注入。

属性说明
constructor通过构造方法进行自动注入,spring会匹配与构造方法参数类型一致的bean进行注入。如果有一个多参数的构造方法,
一个只有一个参数的构造方法,在容器中查找到多个匹配多参数构造方法的bean,那么spring会优先将bean注入到多参数的构造方法中。
byName被注入bean的id名必须与set方法后半截匹配,并且id名称的第一个单词首字母必须小写,这一点与手动set注入有点不同。
byType查找所有的set方法,将符合符合参数类型的bean注入。

比如:

@Service
public class UserServiceImpl {@Autowiredprivate UserDaoImpl userDao;public List<User> findUserList() {return userDao.findUserList();}
}

小结

  其实IoC对编程带来的最大改变不是从代码上,而是从思想上,发生了“主从换位”的变化。应用程序原本是老大,要获取什么资源都是主动出击,但是在IoC/DI思想中,应用程序就变成被动的了,被动的等待IoC容器来创建并注入它所需要的资源了。IoC很好的体现了面向对象设计法则之一—— 好莱坞法则:“别找我们,我们找你”;即由IoC容器帮对象找相应的依赖对象并注入,而不是由对象主动去找。

在这里插入图片描述

  • https://www.pdai.tech/md/spring/spring-x-framework-ioc.html

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

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

相关文章

吴恩达机器学习笔记:第 9 周-15 异常检测(Anomaly Detection) 15.3-15.4

目录 第 9 周 15、 异常检测(Anomaly Detection)15.3 算法15.4 开发和评价一个异常检测系统 第 9 周 15、 异常检测(Anomaly Detection) 15.3 算法 在本节视频中&#xff0c;我将应用高斯分布开发异常检测算法。 异常检测算法&#xff1a;对于给定的数据集 x ( 1 ) , x ( 2…

经典网络解读——Efficientnet

论文&#xff1a;EfficientNet: Rethinking Model Scaling for Convolutional Neural Networks&#xff08;2019.5&#xff09; 作者&#xff1a;Mingxing Tan, Quoc V. Le 链接&#xff1a;https://arxiv.org/abs/1905.11946 代码&#xff1a;https://github.com/tensorflow/t…

python安卓自动化pyaibote实践------学习通自动刷课

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 本文是一个完成一个自动播放课程&#xff0c;避免人为频繁点击脚本的构思与源码。 加油&#xff01;为实现全部电脑自动化办公而奋斗&#xff01; 为实现摆烂躺平的人生而奋斗&#xff01;&#xff01;&#xff…

【全开源】最新恋爱交友脱单盲盒源码

PHP开源版&#xff0c;带扩列付费恋爱定位入群&#xff0c;内有详细安装教程&#xff0c;轻松部署&#xff0c;搭建即可运营&#xff0c;内置永久免费更新地址&#xff0c;后续无忧升级。 程序介绍&#xff1a; 近期爆火的模式&#xff0c;无压力付费交友&#xff0c;由线下摆…

调用WinPE给现有的Windows做一个备份

前言 前段时间有小伙伴问我&#xff1a;如何让给电脑备份系统。 小白直接告诉他&#xff1a;为啥要备份系统呢&#xff1f;直接给电脑创建一个还原点就好了。 Windows还原点创建教程&#xff08;点我跳转&#xff09; 没想到小伙伴的格局比小白大得多&#xff0c;他说&…

C语言----函数

1.函数的概念 函数&#xff1a;founction c语言的程序代码都是函数组成的 c语言中的函数就是一个完成某项特定的任务的一段代码&#xff0c;这段代码有特殊的写法和调用方法 c语言中我们一般见到两种函数&#xff1a; .库函数 .自定义函数 2.库函数 有对应的头文件 #i…

AutoCAD 2025 for mac/win:设计未来,触手可及

在数字化时代&#xff0c;设计不再局限于纸笔之间&#xff0c;而是跃然于屏幕之上&#xff0c;AutoCAD 2025正是这一变革的杰出代表。无论是Mac用户还是Windows用户&#xff0c;AutoCAD 2025都以其卓越的性能和出色的用户体验&#xff0c;成为了CAD设计绘图领域的佼佼者。 Aut…

什么是 Web3 的生成式 AI?

从 Web 1.0 的静态、单向通信到 Web 2.0 的动态、用户驱动的格局&#xff0c;互联网在二十年的时间里经历了一场显着的转变。现在&#xff0c;当我们站在 Web 3.0 时代的边缘时&#xff0c;我们正在见证更具颠覆性的事物的曙光&#xff1a;生成式人工智能 (AI) 融入我们的数字世…

4月28日,深圳Sui Meetup活动圆满成功

对于Sui来说&#xff0c;2024年无疑是充满历史意义的一年。在这几个月的时间里&#xff0c;Sui凭借其革命性的技术架构和稳固的生态系统&#xff0c;在区块链界中如同新星般冉冉升起。 其总锁定价值&#xff08;TVL&#xff09;屡创新高&#xff0c;链上生态系统繁荣昌盛&…

【Python小练】求斐波那契数列第n个数

题目 输出斐波那契数列第n个数。 分析 首先我们要知道&#xff0c;斐波那契数列&#xff0c;这个数列从第三位开始等于前两个数的和&#xff0c;要知道数列第n个数&#xff08;n>2&#xff09;&#xff0c;就要知道其前两相的值&#xff0c;着就需要用到递归了。来看一下吧…

平面模型上提取凸凹多边形------pcl

平面模型上提取凸凹多边形 pcl::PointCloud<pcl::PointXYZ>::Ptr PclTool::ExtractConvexConcavePolygons(pcl::PointCloud<pcl::PointXYZ>::Ptr cloud) {pcl::PointCloud<pcl::PointXYZ>::Ptr cloud_filtered(new pcl::PointCloud<pcl::PointXYZ>);p…

Windows如何通过wsl2迅速启动Docker desktop的PHP的Hyperf项目容器?

一、安装WSL 什么是WSL&#xff1f; 官网&#xff1a;什么是WSL&#xff1f; Windows Subsystem for Linux (WSL) 是一个在Windows 10和Windows 11上运行原生Linux二进制可执行文件的兼容性层。 换句话说&#xff0c;WSL让你可以在Windows系统上运行Linux环境&#xff0c;而无需…