IOC也是Spring的核心之一了,之前学的时候是采用xml配置文件的方式去实现的,后来其中也多少穿插了几个注解,但是没有说完全采用注解实现。那么这篇文章就和大家分享一下,全部采用注解来实现IOC+DI
1. Spring 是什么
我们通常所说的 Spring 指的是 Spring Framework(Spring 框架),它是一个开源框架,有着活跃而庞 大的社区,这就是它之所以能长久不衰的原因。Spring 支持广泛的应用场景,它可以让 Java 企业级的应用程序开发起来更简单。
一句话概括 Spring :Spring 是包含了众多工具方法的 IOC 容器
那么什么是 IOC 呢
2. IOC 是什么
IOC (Inversion of Control 控制反转),就是说 Spring 是一个控制反转的容器
既然提到了控制反转,那么就来看看什么是正向控制,什么是反向控制
2.1 正向控制
在我们之前的大部分代码中,当我们要用到一个对象时,我们就需要自己构造这个对象,自己 new 对象,自己造,自己用,典型的自给自足生产模式
2.2 反向控制
在 Spring 的大部分代码中,我们只需要声明我们需要一个什么样的对象,对象的构造我们不管,这里的对象就是 Spring 帮我们注入进来的,对象就来自于 IOC 容器
3. DI 概念说明
说到 IoC 不得不提的一个词就是“DI”,DI 是 Dependency Injection 的缩写,翻译成中文是“依赖注入”的 意思。
所谓依赖注入,就是由 IoC 容器在运行期间,动态地将某种依赖关系注入到对象之中。所以,依赖注入 (DI)和控制反转(IoC)是从不同的角度的描述的同一件事情,就是指通过引入 IoC 容器,利用依赖关系注入的方式,实现对象之间的解耦。
4. Spring 的工作模式可以理解成一个买卖对象的市场
Spring 容器就是这个市场,我们既可以是卖家,也可以是买家
当我们是卖家时,将我们的对象注入到 Spring 库
当我们是买家时,声明我们需要的对象,Spring 从库中将对象取出来给我们
5. 以一个小程序来理解 IOC 和 DI
当我们需要造一辆车时,就需要得到一个四个部分,才能造成一辆车
以代码实现时,我们就需要这四个对象,先手动将这四个对象注入到库中
@Component:将被修饰的类注册到库中
@Autowired:自动注入,在这里可以理解成从库中拿到所修饰的元素
这里重写 toSpring() 只是为了打印对象,没有其他特殊意义
使用 context.getBeanDefinitionNames() 得到库中所有对象并打印后,可以清楚的看到,这四个对象已经被注入到库中
后续 car 对象也成功打印
@Component
public class Car {
@Autowired
private Framework framework;
@Override
public String toString() {
return "Car{" +
"framework=" + framework +
'}';
}
}
@Component
public class Framework {
@Autowired
private Bottom bottom;
@Override
public String toString() {
return "Framework{" +
"bottom=" + bottom +
'}';
}
}
@Component
public class Bottom {
@Autowired
private Tire t1;
@Autowired
private Tire t2;
@Autowired
private Tire t3;
@Autowired
private Tire t4;
@Override
public String toString() {
return "Bottom{" +
"t1=" + t1 +
", t2=" + t2 +
", t3=" + t3 +
", t4=" + t4 +
'}';
}
}
@Component
public class Tire {
@Override
public String toString() {
return "轮子";
}
}
主类
@SpringBootApplication
public class IocApplication {
public static void main(String[] args) {
ConfigurableApplicationContext context = SpringApplication.run(IocApplication.class, args);
Car car = context.getBean(Car.class);
// 得到库中所有的 bean(对象)
String[] beanDefinitionNames = context.getBeanDefinitionNames();
for (String beanDefinitionName : beanDefinitionNames) {
System.out.println(beanDefinitionName);
}
System.out.println(car);
}
}
当有一个对象没有被 @Component 修饰,即库中没有这个对象,那么其他对象就不能用@Autowired 得到这个对象,则就会出现以下异常
6. 以 XML 文件的方式将对象注入到我们自己创建的库中
6.1 新建一个 XML 文件
现在这个 XML 文件就可以看成是一个空的 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/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
</beans>
6.2 在 Main 中操作这个库
用 FileSystemXmlApplicationContext() 打开这个 XML 文件之后,就可以在这里对刚刚创建的 Spring 进行一些操作
public class Main {
public static void main(String[] args) {
//1. 以 XML 配置文件手动创建 Spring 容器
FileSystemXmlApplicationContext context = new FileSystemXmlApplicationContext("spring-config.xml");
// 2. 打印容器中所有对象的 bean-id
String[] names = context.getBeanDefinitionNames();
System.out.println("所有对象陈列:");
for (String name : names) {
Object bean = context.getBean(name);
// 得到此类的全名,即包含包名的名称
String canonicalName = bean.getClass().getCanonicalName();
System.out.println(name + " " + canonicalName);
}
int count = context.getBeanDefinitionCount();
System.out.println("一共有" + count + "个");
}
}
6.3 XML 文件的方式注入 bean(对象)
以 bean 标签来对对象进行注入,以下 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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<!-- xml的方式注入类,new 的过程也交给 Spring-->
<bean id="empty-class" class="com.hsq.ioc2.EmptyClass"/>
<!-- 构造方法注入类-->
<bean id="second-class" class="com.hsq.ioc2.SecondClass">
<!-- 依赖 EmptyClass 注入 SecondClass-->
<constructor-arg ref="empty-class"/>
</bean>
<!-- set 方法注入类-->
<bean id="third-class" class="com.hsq.ioc2.ThirdClass">
<property name="EmptyClass" ref="empty-class"/>
<property name="SecondClass" ref="second-class"/>
</bean>
</beans>
public class EmptyClass {
public EmptyClass() {
System.out.println("被调用");
}
}
public class SecondClass {
public SecondClass(EmptyClass emptyClass) {
System.out.println("second 被调用");
}
}
public class ThirdClass {
public EmptyClass emptyClass;
public SecondClass secondClass;
public void setEmptyClass(EmptyClass emptyClass) {
System.out.println("setEmptyClass() " + emptyClass);
this.emptyClass = emptyClass;
}
public void setSecondClass(SecondClass secondClass) {
System.out.println("setSecondClass() " + secondClass);
this.secondClass = secondClass;
}
public ThirdClass() {
System.out.println("third 被调用");
}
}
到此这篇关于Spring IOC与DI核心重点分析的文章就介绍到这了,更多相关Spring IOC与DI内容请搜索编程学习网以前的文章希望大家以后多多支持编程学习网!
本文标题为:Spring IOC与DI核心重点分析
基础教程推荐
- Java实现查找文件和替换文件内容 2023-04-06
- Java并发编程进阶之线程控制篇 2023-03-07
- JDK数组阻塞队列源码深入分析总结 2023-04-18
- Java数据结构之对象比较详解 2023-03-07
- java基础知识之FileInputStream流的使用 2023-08-11
- springboot自定义starter方法及注解实例 2023-03-31
- ConditionalOnProperty配置swagger不生效问题及解决 2023-01-02
- Java文件管理操作的知识点整理 2023-05-19
- Java实现线程插队的示例代码 2022-09-03
- java实现多人聊天系统 2023-05-19