更多内容请查看笔记目录:【Spring 5.x】学习笔记汇总
什么是对象的⽣命周期?
为什么要学习对象的⽣命周期?
⽣命周期的 3 个阶段:
Spring 工厂何时创建对象?
scope="prototype"
:Spring 工厂在获取对象 ctx.getBean("xxx")
的同时,创建对象。scope="singleton"
:Spring 工厂创建的同时,创建对象。
也可以实现工厂获取对象的同时,创建对象。什么时候?Spring 工厂在创建完对象后,调用对象的初始化方法,完成对应的初始化操作。
初始化方法提供:程序员根据需求,提供初始化方法,最终完成初始化操作。
初始化方法调用:Spring 工厂进行调用。
提供初始化方法的两种方式:
InitializingBean
接口:public class Product implements InitializingBean {
//程序员根据需求实现的方法, 完成初始化操作
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("Product.afterPropertiesSet");
}
}
init-method
:public class Product {
public void myInit() {
System.out.println("Product.myInit");
}
}
<bean id="product" class="com.yusael.life.Product" init-method="myInit"/>
初始化操作的细节分析:
如果⼀个对象既实现 InitializingBean
同时⼜提供的 普通的初始化方法,执行顺序?
先执行 InitializingBean
,再执行 普通初始化方法。
注入⼀定发⽣在初始化操作的前面。
初始化操作到底是什么?
资源的初始化:数据库、IO、网络、…
Spring 销毁对象前,会调用对象的销毁方法,完成销毁操作。
Spring 什么时候销毁所创建的对象?ctx.close();
销毁方法提供:程序员根据业务需求,定义销毁方法,完成销毁操作
销毁方法调用:Spring 工厂进行调用。
开发流程与初始化操作一样,提供销毁方法的两种方式:
DisposableBean
接口:public class Product implements DisposableBean {
// 程序员根据⾃⼰的需求, 定义销毁方法, 完成销毁操作
@Override
public void destroy() throws Exception {
System.out.println("Product.destroy");
}
}
destroy-method
:public class Product {
// 程序员根据⾃⼰的需求, 定义销毁方法, 完成销毁操作
public void myDestory() {
System.out.println("Product.myDestory");
}
}
<bean id="product" class="com.yusael.life.Product" destroy-method="myDestory"/>
销毁阶段细节分析:
scope="singleton"
,初始化操作都适用。io.close()
、connection.close()
、…public class Product implements InitializingBean, DisposableBean {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
System.out.println("Product.setName");
this.name = name;
}
Product() { // 创建
System.out.println("Product.Product");
}
// 程序员根据需求实现的方法, 完成初始化操作
public void myInit() {
System.out.println("Product.myInit");
}
// 程序员根据需求实现的方法, 完成初始化操作
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("Product.afterPropertiesSet");
}
public void myDestory() {
System.out.println("Product.myDestory");
}
// 程序员根据⾃⼰的需求, 定义销毁方法, 完成销毁操作
@Override
public void destroy() throws Exception {
System.out.println("Product.destroy");
}
}
<bean id="product" class="com.yusael.life.Product" init-method="myInit" destroy-method="myDestory">
<property name="name" value="yusael"/>
bean>
配置文件参数化:把 Spring 配置文件中需要经常修改的字符串信息,转移到⼀个更小的配置文件中。
优点:利于 Spring 配置文件的维护(修改)
jdbc.driverClassName = com.mysql.jdbc.Driver
jdbc.url = jdbc:mysql://localhost:3306/spring?useSSL=false
jdbc.username = root
jdbc.password = 1234
<context:property-placeholder location="classpath:/db.properties"/>
在 Spring 配置文件中通过 ${key}
获取小配置文件中对应的值:
<bean id="conn" class="com.yusael.factorybean.ConnectionFactoryBean">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
bean>
作用:Spring 通过 类型转换器 把 配置文件 中 字符串 类型的数据,转换成了对象中成员变量对应类型的数据,进而完成了注入。
产生原因:当 Spring 内部没有提供特定类型转换器时,而程序员在应用的过程中还需要使用,那么
就需要程序员⾃⼰定义类型转换器。
[开发步骤]:
public class MyDateConverter implements Converter<String, Date> {
/*
convert方法作用: String ---> Date
SimpleDateFormat sdf = new SimpleDateFormat();
sdf.parset(String) ---> Date
参数:
source : 代表的是配置文件中, 日期字符串 2020-10-11
return : 当把转换好的 Date 作为 convert 方法的返回值后,
Spring ⾃动的为birthday属性进行注入(赋值)
*/
@Override
public Date convert(String source) {
Date date = null;
try {
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
date = sdf.parse(source);
} catch (ParseException e) {
e.printStackTrace();
}
return date;
}
}
MyDateConverter
对象,再注册类型转换器;
<bean id="myDateConverter" class="com.yusael.converter.MyDateConverter"/>
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<set>
<ref bean="myDateConverter"/>
set>
property>
bean>
<bean id="good" class="com.yusael.converter.Good">
<property name="name" value="zhenyu"/>
<property name="birthday" value="2012-12-12"/>
bean>
MyDateConverter
中的日期的格式,通过 依赖注入 的方式,由配置文件完成赋值。public class MyDateConverter implements Converter<String, Date> {
private String pattern;
@Override
public Date convert(String source) {
Date date = null;
try {
SimpleDateFormat sdf = new SimpleDateFormat(pattern);
date = sdf.parse(source);
} catch (ParseException e) {
e.printStackTrace();
}
return date;
}
public String getPattern() {
return pattern;
}
public void setPattern(String pattern) {
this.pattern = pattern;
}
}
<bean id="myDateConverter" class="com.yusael.converter.MyDateConverter">
<property name="pattern" value="yyyy-MM-dd"/>
bean>
ConversionSeviceFactoryBean
定义 id属性,值必须是 conversionService
;<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<set>
<ref bean="myDateConverter"/>
set>
property>
bean>
2020/05/01
。<bean id="good" class="com.yusael.converter.Good">
<property name="name" value="zhenyu"/>
<property name="birthday" value="2012/12/12"/>
bean>
BeanPostProcessor 作用:对 Spring 工厂所创建的对象,进行再加工。(AOP 的底层实现)
程序员实现 BeanPostProcessor
接口中规定的两个方法:
public Object postProcessBeforeInitialization(Object bean, String beanName) {
return bean;
}
作用:Spring 创建完对象,并进行注入后,可以运行 Before
⽅法进行加工;
public Object postProcessAfterInitialization(Object bean, String beanName) {
return bean;
}
作⽤:Spring 执行完对象的初始化操作后,可以运行 After
⽅法进行加工;
实战中:很少处理 Spring 的初始化操作,没有必要区分 Before
,After
。只需要实现其中一个,建议是 After
方法即可。
BeanPostProcessor
接口public class MyBeanPostProcessor implements BeanPostProcessor {
@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
return null;
}
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
Category category = (Category) bean;
category.setName("yusael");
return category;
}
}
<bean id="myBeanPostProcessor" class="com.yusael.beanpost.MyBeanPostProcessor"/>
细节:BeanPostProcessor 会对 Spring 工厂创建的所有对象进行加工。如果工厂创建了多个不同的对象,要注意区别传入的对象:
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
if (bean instanceof Category) {
Category category = (Category) bean;
category.setName("yusael");
return category;
}
return bean;
}