第2章 Spring核心概念
概要:
- Ioc/DI思想
- Ioc容器
- Bean
代码现状分析:
//业务实现
public class BookServiceImpl implements BookService{
private BookDao bookDao = new BookDaoImpl();
public void save() {
bookDao.save();
}
}
//数据层实现
public class BookDaoImpl implements BookDao{
public void save() {
System.out.println("book dao save...");
}
}
上面是通过 new BookDaoImpl()来实现数据层调用,如果一旦需要更换数据层方法为BookDaoImpl2,比如:
public class BookDaoImpl2 implements BookDao{
public void save() {
System.out.println("book dao save..2.");
}
}
那么上面的 new BookDaoImpl()也得改成 new BookDaoImpl()2,这样 代码耦合性偏高,解决方案如下:
- 将对象的创建控制器有程序转移到外部,这种思想成为控制反转。
📖2.1 Ioc/DI思想
✅什么是Ioc/DI
IoC(Inversion of Control)控制反转:
使用对象时,由主动new产生对象转换为由外部提供对象,此过程中对象创建控制权由程序转移到外部,此思想称为控制反转。
spring技术对Ioc思想进行了实现:
- 提供了一个Ioc容器,用来充当Ioc思想的"外部"。
- Ioc容器负责对象的创建、初始化等一系列工作,被创建或被管理的对象在IoC容器中统称为Bean。
DI(Dependency Injection)依赖注入:
- 在容器中建立bean与bean之间的依赖关系的整个过程,称为依赖注入。
这样做的目的:充分解耦
使用Ioc容器管理bean(Ioc)
在Ioc容器内将有依赖关系的bean进行关系绑定(DI)
最终实现的效果:
- 使用对象时不仅可以直接从Ioc容器中获取,并且获取到的bean已经绑定了所有的依赖关系
✅什么是IOC容器
Spring创建了一个容器用来存放所创建的对象,这个容器就叫IOC容器
✅什么是Bean
spring容器中所存放的一个个对象就叫Bean或Bean对象
<!--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 http://www.springframework.org/schema/beans/spring-beans.xsd">
<!--bean对象-->
<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
</beans>
✅Ioc入门案例(XML版)
需求分析:将BookServiceImpl和BookDaoImpl交给Spring管理,并从容器中获取对应的bean对象进行方法调用。
1.创建Maven的java项目
2.pom.xml添加Spring的依赖jar包
3.创建BookService,BookServiceImpl,BookDao和BookDaoImpl四个类
4.resources下添加spring配置文件,并完成bean的配置
5.使用Spring提供的接口完成IOC容器的创建
6.从容器中获取对象进行方法调用
步骤1:创建Maven项目
步骤2:添加Spring的依赖jar包
pom.xml
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.2.10.RELEASE</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
</dependencies>
步骤3:添加案例中需要的类
创建BookService,BookServiceImpl,BookDao和BookDaoImpl四个类
public interface BookDao {
public void save();
}
public class BookDaoImpl implements BookDao {
public void save() {
System.out.println("book dao save ...");
}
}
public interface BookService {
public void save();
}
public class BookServiceImpl implements BookService {
private BookDao bookDao = new BookDaoImpl();
public void save() {
System.out.println("book service save ...");
bookDao.save();
}
}
步骤4:添加spring配置文件
resources下添加spring配置文件applicationContext.xml,并完成bean的配置
步骤5:在配置文件中完成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">
<!--bean标签标示配置bean
id属性标示给bean起名字
class属性表示给bean定义类型
-->
<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.itheima.service.impl.BookServiceImpl"/>
</beans>
注意事项:bean定义时id属性在同一个上下文中(配置文件)不能重复
步骤6:获取IOC容器
使用Spring提供的接口完成IOC容器的创建,创建App类,编写main方法
public class App {
public static void main(String[] args) {
//获取IOC容器
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
}
}
步骤7:从容器中获取对象进行方法调用
public class App {
public static void main(String[] args) {
//获取IOC容器
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
// BookDao bookDao = (BookDao) ctx.getBean("bookDao");
// bookDao.save();
BookService bookService = (BookService) ctx.getBean("bookService");
bookService.save();
}
}
步骤8:运行程序
测试结果为:
Spring的IOC入门案例已经完成,但是在BookServiceImpl
的类中依然存在BookDaoImpl
对象的new操作,它们之间的耦合度还是比较高,这块该如何解决,就需要用到下面的DI:依赖注入
。
✅DI入门案例(XML版)
需求:基于IOC入门案例,在BookServiceImpl类中删除new对象的方式,使用Spring的DI完成Dao层的注入
1.删除业务层中使用new的方式创建的dao对象
2.在业务层提供BookDao的setter方法
3.在配置文件中添加依赖注入的配置
4.运行程序调用方法
步骤1: 去除代码中的new
在BookServiceImpl类中,删除业务层中使用new的方式创建的dao对象
public class BookServiceImpl implements BookService {
//删除业务层中使用new的方式创建的dao对象
private BookDao bookDao;
public void save() {
System.out.println("book service save ...");
bookDao.save();
}
}
步骤2:为属性提供setter方法
在BookServiceImpl类中,为BookDao提供setter方法
public class BookServiceImpl implements BookService {
//删除业务层中使用new的方式创建的dao对象
private BookDao bookDao;
public void save() {
System.out.println("book service save ...");
bookDao.save();
}
//提供对应的set方法
public void setBookDao(BookDao bookDao) {
this.bookDao = bookDao;
}
}
步骤3:修改配置完成注入
在配置文件中添加依赖注入的配置
<?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">
<!--bean标签标示配置bean
id属性标示给bean起名字
class属性表示给bean定义类型
-->
<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
<!--配置server与dao的关系-->
<!--property标签表示配置当前bean的属性
name属性表示配置哪一个具体的属性
ref属性表示参照哪一个bean
-->
<property name="bookDao" ref="bookDao"/>
</bean>
</beans>
注意:配置中的两个bookDao的含义是不一样的
- name="bookDao"中
bookDao
的作用是让Spring的IOC容器在获取到名称后,将首字母大写,前面加set找对应的setBookDao()
方法进行对象注入 - ref="bookDao"中
bookDao
的作用 让Spring能在IOC容器中找到id为bookDao
的Bean对象给bookService
进行注入 - 综上所述,对应关系如下:
步骤4:运行程序
运行,测试结果为:
📖2.2 Ioc相关内容
✅bean基本配置
bean的基本配置主要下面四种:
属性名称 | 功能 |
---|---|
id | 定义bean的id,在容器中id值唯一 |
class | 定义bean的类型,即配置的bean的全路径类名 |
name | 定义bean的别名,可以定义多个,使用逗号(,)分号(;)空格( )分隔 |
scope | 定义bean的作用范围,默认 singleton 为单例, prototype 为非单例 |
ref | 指定bean,必须在容器中存在,否则会报错 |
下面是一个标准bean容器对象:
<beans>
<bean id="bookService" class="spring.service.Impl.BookServiceImpl"/>
</beans>
name属性:配置别名后可根据容器名称获取bean对象:
<!--name:为bean指定别名,别名可以有多个,使用逗号,分号,空格进行分隔-->
<bean id="bookService" name="service service4 bookEbi" class="com.itheima.service.impl.BookServiceImpl">
</bean>
public class AppForName {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
//此处根据bean标签的id属性和name属性的任意一个值来获取bean对象
BookService bookService = (BookService) ctx.getBean("service4");
bookService.save();
}
}
scope属性:如何要验证bean对象是否为单例,只需同一个bean获取两次,将对象打印到控制台,看打印出的地址值是否一致。
//配置单例
<bean id="bookDao" name="dao" class="com.itheima.dao.impl.BookDaoImpl" scope="singleton"/>
配置非单例
<bean id="bookDao" name="dao" class="com.itheima.dao.impl.BookDaoImpl" scope="prototype"/>
public class AppForScope {
public static void main(String[] args) {
ApplicationContext ctx = new
ClassPathXmlApplicationContext("applicationContext.xml");
BookDao bookDao1 = (BookDao) ctx.getBean("bookDao");
BookDao bookDao2 = (BookDao) ctx.getBean("bookDao");
System.out.println(bookDao1);
System.out.println(bookDao2);
}
}
ref属性:bean依赖注入的ref属性指定bean,必须在容器中存在
如果不存在,则会报错,如下:
这个错误大家需要特别关注下:
获取bean无论是通过id还是name获取,如果无法获取到,将抛出异常NoSuchBeanDefinitionException
✅bean的实例化
Spring的IOC实例化对象的三种方式分别是:
实例化方式 | 属性 |
---|---|
构造方法(常用) | |
静态工厂(了解) | factory-method:指定工厂bean配置 |
实例工厂(了解) | |
FactoryBean(实用) |
🔖构造方法(常用)
1.提供可访问的构造方法
public class BookDaoImpl implements BookDao{
public BookDaoImpl() {
System.out.println("book dao constructor...");
}
public void save() {
System.out.println("book dao save...");
}
}
2.配置:将对应的BookDaoImpl进行bean配置
<bean id="bookDao" class="com.spring.dao.BookDaoImpl"></bean>
注意 :如果没有无参构造方法,将抛出BeanCreationException异常
3.获取bean的方法进行运行测试
public class AppForInstanceBook {
public static void main(String[] args) {
ApplicationContext ctx = new
ClassPathXmlApplicationContext("applicationContext.xml");
BookDao bookDao = (BookDao) ctx.getBean("bookDao");
bookDao.save();
}
}
运行程序,如果控制台有打印构造函数中的输出,说明Spring容器在创建对象的时候也走的是构造函数
🔖静态工厂
1.静态工厂
public class OrderDaoFactory {
public static OrderDao getOrderDao(){
System.out.println("OrderDaoFactory getOrderDao runing...");
return new OrderDaoImpl();
}
}
2.配置bean
<bean id="orderDao" class="com.spring.factory.OrderDaoFactory" factory-method="getOrderDao"></bean>
- class配置的是工厂类
- 必须指定工厂中哪个方法去创建对象
3.获取bean的方法进行运行测试
public class AppForInstanceOrder {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
OrderDao orderDao = (OrderDao) ctx.getBean("orderDao");
orderDao.save();
}
}
🔖实例工厂(了解)
1.一个没有静态修饰的工厂
public class UserDaoFactory {
public UserDao getOrderDao(){
System.out.println("UserDaoFactory getOrderDao running...");
return new UserDaoImpl();
}
}
2.配置
<!--实例化工厂初始化bean-->
<bean id="userDaoFactory" class="com.spring.factory.UserDaoFactory"></bean>
<bean id="userDao" factory-bean="userDaoFactory" factory-method="getOrderDao"></bean>
factory-bean指定工厂bean,factory-method指定返回实例的工厂方法
3.获取对象
public class AppUserFactory {
public static void main(String[] args) {
//spring实例化工厂创建对象
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ctx.getBean("userDao");
userDao.save();
}
}
🔖FactoryBean实例化(实用)
这第四种实例化其实是spring在实例工厂的基础上做了改进,也就是FactoryBean实例化。这种方式在Spring去整合其他框架的时候会被用到,所以这种方式需要理解掌握。
1.创建FactoryBean工厂,需要实现FactoryBean
接口
public class UserDaoFactoryBean implements FactoryBean<UserDao> {
//代替原始实例工厂中创建对象的方法
@Override
public UserDao getObject() throws Exception {
return new UserDaoImpl();
}
@Override
public Class<?> getObjectType() {
return UserDao.class;
}
}
2.配置
<bean id="userDao" class="com.spring.factory.UserDaoFactoryBean"></bean>
3.获取对象
public class AppUserFactoryBean {
public static void main(String[] args) {
//FactoryBean实例化
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ctx.getBean("userDao");
userDao.save();
}
}
查看源码会发现,FactoryBean接口其实会有三个方法,分别是:
T getObject() throws Exception;
Class<?> getObjectType();
default boolean isSingleton() {
return true;
}
方法一:getObject(),被重写后,在方法中进行对象的创建并返回
方法二:getObjectType(),被重写后,主要返回的是被创建类的Class对象
方法三:没有被重写,因为它已经给了默认值,从方法名中可以看出其作用是设置对象是否为单例,默认true,从意思上来看,我们猜想默认应该是单例,如何来验证呢?
✅bean的生命周期
bean属性 | 作用 |
---|---|
init-method | 定义bean的初始化方法 |
destroy-method | 定义bean的销毁方法 |
🔖生命周期设置
添加生命周期的控制方法,具体的控制有两个阶段:
- bean创建之后,想要添加内容,比如用来初始化需要用到资源
- bean销毁之前,想要添加内容,比如用来释放用到的资源
步骤1:添加初始化和销毁方法
针对这两个阶段,我们在BooDaoImpl类中分别添加两个方法,方法名任意
public class BookDaoImpl implements BookDao {
public void save() {
System.out.println("book dao save ...");
}
//表示bean初始化对应的操作
public void init(){
System.out.println("init...");
}
//表示bean销毁前对应的操作
public void destory(){
System.out.println("destory...");
}
}
步骤2:配置生命周期
在配置文件添加配置,如下:
<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl" init-method="init" destroy-method="destory"/>
步骤3:运行程序
运行AppForLifeCycle打印结果为:
从结果中可以看出,init方法执行了,但是destroy方法却未执行,这是为什么呢?
- Spring的IOC容器是运行在JVM中
- 运行main方法后,JVM启动,Spring加载配置文件生成IOC容器,从容器获取bean对象,然后调方法执行
- main方法执行完后,JVM退出,这个时候IOC容器中的bean还没有来得及销毁就已经结束了
- 所以没有调用对应的destroy方法
🔖close关闭容器
ApplicationContext中没有close方法
需要将ApplicationContext更换成ClassPathXmlApplicationContext
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
调用ctx的close()方法
ctx.close();
运行程序,就能执行destroy方法的内容
🔖注册钩子关闭容器
在容器未关闭之前,提前设置好回调函数,让JVM在退出之前回调此函数来关闭容器
调用ctx的registerShutdownHook()方法
ctx.registerShutdownHook();
注意:
registerShutdownHook
在ApplicationContext中也没有运行后,查询打印结果
两种方式介绍完后,close
和registerShutdownHook
选哪个?
相同点:这两种都能用来关闭容器
不同点:close()是在调用的时候关闭,registerShutdownHook()是在JVM退出前调用关闭。
分析上面的实现过程,会发现添加初始化和销毁方法,即需要编码也需要配置,实现起来步骤比较多也比较乱。
Spring提供了两个接口来完成生命周期的控制,好处是可以不用再进行配置init-method
和destroy-method
接下来在BookServiceImpl完成这两个接口的使用:
修改BookServiceImpl类,添加两个接口InitializingBean
, DisposableBean
并实现接口中的两个方法afterPropertiesSet
和destroy
public class BookServiceImpl implements BookService, InitializingBean, DisposableBean {
private BookDao bookDao;
public void setBookDao(BookDao bookDao) {
this.bookDao = bookDao;
}
public void save() {
System.out.println("book service save ...");
bookDao.save();
}
public void destroy() throws Exception {
System.out.println("service destroy");
}
public void afterPropertiesSet() throws Exception {
System.out.println("service init");
}
}
重新运行AppForLifeCycle类,
那第二种方式的实现,我们也介绍完了。
小细节
对于InitializingBean接口中的afterPropertiesSet方法,翻译过来为
属性设置之后
。对于BookServiceImpl来说,bookDao是它的一个属性
setBookDao方法是Spring的IOC容器为其注入属性的方法
思考:afterPropertiesSet和setBookDao谁先执行?
从方法名分析,猜想应该是setBookDao方法先执行
验证思路,在setBookDao方法中添加一句话
public void setBookDao(BookDao bookDao) { System.out.println("set ....."); this.bookDao = bookDao; }
重新运行AppForLifeCycle,打印结果如下:
验证的结果和我们猜想的结果是一致的,所以初始化方法会在类中属性设置之后执行。
🔖bean生命周期小结
(1)关于Spring中对bean生命周期控制提供了两种方式:
- 在配置文件中的bean标签中添加
init-method
和destroy-method
属性 - 类实现
InitializingBean
与DisposableBean
接口,这种方式了解下即可。
(2)对于bean的生命周期控制在bean的整个生命周期中所处的位置如下:
- 初始化容器
- 1.创建对象(内存分配)
- 2.执行构造方法
- 3.执行属性注入(set操作)
- 4.执行bean初始化方法
- 使用bean
- 1.执行业务操作
- 关闭/销毁容器
- 1.执行bean销毁方法
(3)关闭容器的两种方式:
- ConfigurableApplicationContext是ApplicationContext的子类
- close()方法
- registerShutdownHook()方法
📖2.3 DI相关内容
思考:向一个类中传递数据的方式有几种?
- 普通方法(set方法)
- 构造方法
思考:依赖注入描述了在容器中建立bean与bean之间依赖关系的过程,如果bean运行需要的是数字或字符串呢?
- 引用类型
- 简单类型(基本数据类型与String)
所以依赖注入方式可以分为下面几种:
- setter注入
- 简单类型
- 引用类型
- 构造器注入
- 简单类型
- 引用类型
✅Setter注入
setter注入一共两步:
- 对象需提供可访问的
set
方法,set的类型可以是简单数据类型或引用数据类型 - 在配置中使用
property
标签,name
指定set方法中的属性,ref
实现引用类型注入,value
实现简单数据类型注入。
🔖1.简单类型
在配置中使用property
标签,value
属性来实现注入简单数据类型
public class BookDaoImpl implements BookDao {
private int connectionNum;
private String databaseName;
public void setConnectionNum(int connectionNum) {
this.connectionNum = connectionNum;
}
public void setDatabaseName(String databaseName) {
this.databaseName = databaseName;
}
public void save() {
System.out.println("BookDao save ... connectionNum="+connectionNum+",databaseName="+databaseName);
}
}
<bean id="bookDao" class="com.spring.dao.Impl.BookDaoImpl" init-method="init" destroy-method="destroy" >
<property name="databaseName" value="mysql"></property>
<property name="connectionNum" value="100"></property>
</bean>
🔖2.引用类型
引用类型的setter注入在上面将DI入门案例的时候写过,是在配置中使用property
标签,ref
属性来关联bean之间的依赖关系,实现注入引用类型
public class BookServiceImpl implements BookService {
private BookDao bookDao;
public void save() {
System.out.println("service save");
bookDao.save();
}
//set注入bookDao
public void setBookDao(BookDaoImpl bookDao) {
this.bookDao = bookDao;
}
}
<bean id="bookDao" class="com.jdk8.dao.BookDaoImpl"/>
<bean id="bookService" class="com.jdk8.service.BookServiceImpl">
<!--配置service与dao的关系-->
<property name="bookDao" ref="bookDao"/>
</bean>
✅构造器注入
构造器注入和setter注入很相似,只需将原来的set方法换成构造方法即可,也是两步:
- 对象需提供有参的构造方法,参数可以是简单数据类型或引用数据类型
- 在配置中使用
property
标签,name
指定set方法中的属性,ref
实现引用类型注入,value
实现简单数据类型注入。
构造器参数适配(了解):
- type属性设置按形参类型注入
- index属性设置按形参位置注入
🔖1.简单类型
public class BookDaoImpl implements BookDao {
private int connectionNum;
private String databaseName;
// 构造器注入
public BookDaoImpl(int connectionNum, String databaseName) {
this.connectionNum = connectionNum;
this.databaseName = databaseName;
}
public void save() {
System.out.println("BookDao save ... connectionNum="+connectionNum+",databaseName="+databaseName);
}
}
<bean id="bookDao" class="spring.dao.Impl.BookDaoImpl">
<constructor-arg name="connectionNum" value="10"/>
<constructor-arg name="databaseName" value="mysql"/>
</bean>
🔖2.引用类型
public class BookServiceImpl implements BookService {
private BookDao bookDao;
// 构造器注入
public BookServiceImpl(BookDao bookDao) {
this.bookDao = bookDao;
}
public void save() {
System.out.println("book service");
bookDao.save();
}
}
<bean id="bookDao" class="com.jdk8.dao.BookDaoImpl"/>
<bean id="bookService" class="spring.service.Impl.BookServiceImpl">
<constructor-arg name="bookDao" ref="bookDao"/>
</bean>
构造器参数适配(了解即可):解决配置文件与构造器形参耦合度
- 在constructor-arg标签中使用type属性设置按形参类型注入
<bean id="bookDao" class="spring.dao.Impl.BookDaoImpl">
<constructor-arg type="int" value="10"/>
<constructor-arg type="java.lang.String" value="mysql"/>
</bean>
- 在constructor-arg标签中使用index属性设置按形参位置注入
<bean id="bookDao" class="spring.dao.Impl.BookDaoImpl">
<constructor-arg index="0" value="10"/>
<constructor-arg index="1" value="mysql"/>
</bea
依赖注入方式选择
- 建议使用setter注入
- 第三方技术根据情况选择
✅集合注入
- 数组
- List
- Set
- Map
- Properties
public class BookDaoImpl implements BookDao {
private int[] array;
private List<String> list;
private Set<String> set;
private Map<String,String> map;
private Properties properties;
// set
public void setArray(int[] array) {
this.array = array;
}
public void setList(List<String> list) {
this.list = list;
}
public void setSet(Set<String> set) {
this.set = set;
}
public void setMap(Map<String, String> map) {
this.map = map;
}
public void setProperties(Properties properties) {
this.properties = properties;
}
public void save() {
System.out.println("BookDao save ... ");
System.out.println("遍历数组:"+ Arrays.toString(array));
System.out.println("遍历集合:"+ list);
System.out.println("遍历set:"+ set);
System.out.println("遍历map:"+ map);
System.out.println("遍历properties:"+ properties);
}
}
<bean id="bookDao" class="com.spring.dao.Impl.BookDaoImpl">
<!--数组-->
<property name="array">
<array>
<value>100</value>
<value>200</value>
<value>300</value>
</array>
</property>
<!--list-->
<property name="list">
<list>
<value>zhang</value>
<value>li</value>
<value>zeng</value>
</list>
</property>
<!--set-->
<property name="set">
<set>
<value>zhang</value>
<value>li</value>
<value>li</value>
<value>zeng</value>
</set>
</property>
<!--map-->
<property name="map">
<map>
<entry key="zhang" value="18"/>
<entry key="li" value="22"/>
<entry key="zeng" value="25"/>
</map>
</property>
<!--properties-->
<property name="properties">
<props>
<prop key="zhang">18</prop>
<prop key="li">20</prop>
<prop key="zeng">25</prop>
</props>
</property>
</bean>
✅自动装配
什么是依赖自动装配?
Ioc容器根据bean所依赖的资源在容器中查找并注入到bean中的过程称为自动装配。
自动装配的方式:
- 按类型(常用)
- 按名称
- 按构造方法
- 不启用自动装配
🔖按类型(常用)
在bean中设置autoware属性为byType
public class BookServiceImpl implements BookService {
private BookDao bookDao;
// set注入
public void setBookDao(BookDao bookDao) {
this.bookDao = bookDao;
}
public void save() {
System.out.println("book service");
bookDao.save();
}
}
<bean id="bookService" class="com.spring.service.Impl.BookServiceImpl" autowire="byType"/>
需注意两点:
- 对象中的set方法必须有,否则将找不到
- 如果相同类型有多个则报错,可按名称,如下
<bean id="bookDao" class="com.spring.dao.Impl.BookDaoImpl"/>
<bean id="bookDao2" class="com.spring.dao.Impl.BookDaoImpl"/>
🔖按名称
在bean中设置autoware属性为byName
<bean id="bookService" class="com.spring.service.Impl.BookServiceImpl" autowire="byName"/>
🔖依赖自动装配特征
- 自动装配用于引用类型依赖注入,不能对简单类型进行操作
- 使用按类型装配时(byType)必须保障容器中相同类型的bean唯一,推荐使用
- 使用按名称装配时(byName)必须保障容器中具有指定名称的bean,因变量名与配置耦合,不推荐使用
- 自动装配优先级低于setteri注入与构造器注入,同时出现时自动装配配置失效
📖2.4 IOC/DI配置管理第三方bean
✅数据源Druid管理
druid
的依赖
步骤1:导入pom.xml中添加依赖
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.1.16</version>
</dependency>
步骤2:配置第三方bean
在applicationContext.xml配置文件中添加DruidDataSource
的配置
<?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">
<!--管理DruidDataSource对象-->
<bean class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/spring_db"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
</bean>
</beans>
说明:
- driverClassName:数据库驱动
- url:数据库连接地址
- username:数据库连接用户名
- password:数据库连接密码
- 数据库连接的四要素要和自己使用的数据库信息一致。
步骤3:从IOC容器中获取对应的bean对象
public class App {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
DataSource dataSource = (DataSource) ctx.getBean("dataSource");
System.out.println(dataSource);
}
}
步骤4:运行程序
打印如下结果: 说明第三方bean对象已经被spring的IOC容器进行管理
做完案例后,我们可以将刚才思考的两个问题答案说下:
第三方的类指的是什么?
DruidDataSource
如何注入数据库连接四要素?
setter注入
✅数据源C3P0管理
C3P0
的依赖
步骤1:导入pom.xml中添加依赖
<dependency>
<groupId>c3p0</groupId>
<artifactId>c3p0</artifactId>
<version>0.9.1.2</version>
</dependency>
对于新的技术,不知道具体的坐标该如何查找?
直接百度搜索
从mvn的仓库
https://mvnrepository.com/
中进行搜索
步骤2:配置第三方bean
在applicationContext.xml配置文件中添加配置
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"/>
<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/spring_db"/>
<property name="user" value="root"/>
<property name="password" value="root"/>
<property name="maxPoolSize" value="1000"/>
</bean>
注意:
- ComboPooledDataSource的属性是通过setter方式进行注入
- 想注入属性就需要在ComboPooledDataSource类或其上层类中有提供属性对应的setter方法
- C3P0的四个属性和Druid的四个属性是不一样的
步骤3:运行程序
程序会报错,错误如下
报的错为ClassNotFoundException,翻译出来是类没有发现的异常
,具体的类为com.mysql.jdbc.Driver
。错误的原因是缺少mysql的驱动包。
分析出错误的原因,具体的解决方案就比较简单,只需要在pom.xml把驱动包引入即可。
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.47</version>
</dependency>
添加完mysql的驱动包以后,再次运行App,就可以打印出结果:
注意:
- 数据连接池在配置属性的时候,除了可以注入数据库连接四要素外还可以配置很多其他的属性,具体都有哪些属性用到的时候再去查,一般配置基础的四个,其他都有自己的默认值
- Druid和C3P0在没有导入mysql驱动包的前提下,一个没报错一个报错,说明Druid在初始化的时候没有去加载驱动,而C3P0刚好相反
- Druid程序运行虽然没有报错,但是当调用DruidDataSource的getConnection()方法获取连接的时候,也会报找不到驱动类的错误
✅加载properties文件
🔖开启命名空间并加载properties
- 开启context命名空间
- 使用context命名空间,加载指定properties文件
- 使用${}读取加载的属性值
#jdbc.properties配置文件
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/spring_03_database
jdbc.username=root
jdbc.password=123456
<!-- 导入外部属性properties文件 -->
<context:property-placeholder location="jdbc.properties"/>
<!-- 配置数据源,使用${}占位符读取配置文件数据 -->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="${jdbc.driver}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
在context标签中可以设置system-properties-mode
属性来控制是否允许通过系统属性来覆盖属性占位符(Placeholder)中的值。当没有设置system-properties-mode
时会优先从系统变量获取。
<context:property-placeholder location="jdbc.properties" system-properties-mode="NEVER"/>
参数 | 作用 |
---|---|
NEVER | 不允许系统属性覆盖属性占位符中的值 |
FALLBACK | 表示如果属性占位符中的值在系统属性中没有找到对应的键值,则会尝试从系统属性中获取相同键的值来替换属性占位符中的值。 |
OVERRIDE | 表示系统属性会覆盖属性占位符中的任何值,即使属性占位符中已经设置了一个值。 |
🔖加载properties方式
- 不加载系统属性
<context:property-placeholder location="jdbc.properties" system-properties-mode="NEVER"/>
- 加载多个properties文件
<context:property-placeholder location="jdbc.properties,jdbc.properties2" system-properties-mode="NEVER"/>
- 加载所有properties文件
<context:property-placeholder location="*.properties" />
- 加载properties文件标准格式
<context:property-placeholder location="classpath:*.properties" />
- 从类路径或jar包中搜索并加载properties文件
<context:property-placeholder location="classpath*:*.properties" />
📖2.5 核心容器
✅创建容器(了解)
- 方式一:类路径加载配置文件
ApplicationContext ctx new ClassPathXmlApplicationContext("applicationContext.xml");
- 方式二:文件路径加载配置文件
ApplicationContext ctx new FileSystemXmlApplicationContext("D:\\applicationContext.xml");
- 加载多个配置文件
ApplicationContext ctx new ClassPathXmlApplicationContext("bean1.xml","bean2.xml");
✅获取Bean
- 方式一:使用bean名称获取
BookDao bookDao (BookDao)ctx.getBean("bookDao");
- 方式二:使用bean名称获取并指定类型
BookDao bookDao ctx.getBean("bookDao",BookDao.class);
- 方式三:使用bean类型获取
BookDao bookDao ctx.getBean(BookDao.class);
✅容器类层次结构图
✅BeanFactory初始化(了解)
- 类路径加载配置文件
Resource resources = new ClassPathResource("applicationContext.xml");
BeanFactory bf = new XmlBeanFactory(resources);
BookDao bookDao bf.getBean("bookDao",BookDao.class);
bookDao.save();
- BeanFactoryt创建完毕后,所有的bean均为延迟加载
✅核心容器总结
1.容器相关
- BeanFactory是IoC容器的顶层接口,初始化BeanFactory对象时,加载的bean延迟加载
- ApplicationContext:接口是Spring容器的核心接口,初始化时bean立即加载
- ApplicationContext:接口提供基础的bean操作相关方法,通过其他接口扩展其功能
- ApplicationContext接口常用初始化类
- ClassPathXmlApplicationContext
- FileSystemXmlApplicationContext
2.bean相关
3.依赖注入相关