Spring(下)

接上篇,从第八个问题讲起

八.Spring工厂创建复杂对象

1.什么是复杂对象

简单对象就是可以直接new出来的,也就是直接调用构造方法创建

所以复杂对象就是不能直接通过调用构造方法创建。就比如JDBC中的Connection

2.三种方法

(1).FactoryBean接口

首先准备一个类,实现上述接口,进行文件配置。里面有三个方法:

getObject方法是用于书写创建复杂对象的代码,并将复杂对象作为方法返回值返回

getObjectType方法是返回复杂对象的class对象

isSingleton方法:若该对象只需创建一次(就是说每次都是用同一个对象),就让这个方法返回true;若每次使用都要创建一个新的对象,就让它返回false。

示例:

我们拿数据库建立连接来举例。既然要使用数据库,我们就要先引入数据库的依赖。

首先来实现getObject方法

其次就是getObjectType方法:

最后就是isSingleton方法

进行文件配置:

获取连接对象:

如上,虽然在配置文件中指定的是FactoryBean接口的实现类,但是获取到的却是Connection对象!!!这就是为啥要实现factorybean接口,原理会在后面讲到

细节分析:
如果就想获得factoryBean对象

在getbean方法中的conn前面加一个&即可

isSingleton方法

如果返回true,没那么每次创建的对象都是同一个:

如果返回false,则不是同一个:

依赖注入

对于connectionFactoryBean来说,下面几个内容是很重要的,是它所依赖的

所以要对他们进行set注入:

首先设置成成员变量,然后提供set方法,再去配置文件

FactoryBean实现原理

首先根据getBean传入的conn字符串获取对应bean标签的相关信息,

其次使用instanceof判断是否是factorybean的实现类

如果是就去执行重写的getObject方法来获取到connection对象,最后返回对象

总结

FactoryBean是Spring种用于创建复杂对象的方式,也是Spring原生提供的

(2).实例工厂

优势

实例工厂能够避免Spring框架的侵入,避免使用Spring提供的FactoryBean。

实力工厂也可以整合遗留系统

示例

写一个ConnectionFactory类作为遗留系统,

然后再配置文件中进行如下配置:

用到了factory-bean属性和factory-method属性

这样就把遗留系统整合了

(3).静态工厂

将getConnection方法换成静态的

配置文件如上

九.控制对象创建的次数

1.简单对象

在bean标签上加一个scope标签,值为singleton则只能创建一个对象,值为prototype(译为原型)则每次都创建新的对象

scope的值默认为singleton

2.复杂对象

就是使用issingleton方法。如果没有重写这个方法,那就还是默认使用scope属性,并且默认为singleton

3.为啥要控制?

有些对象,比如userDao,就可以被共用,有些不可以

好处:节省不必要的内存浪费

十.对象的生命周期

共分为三个阶段

1.创建阶段

Spring工厂何时创建对象?

当scope为singleton时,就是在Spring工厂创建的同时(也就是new ClassPathXmlApplicationContext)进行对象的创建

当scope为prototype时,就是在获取对象时(getBean方法)进行创建

如何证明?我们在person的无参构造中加一个控制台输出

先指定scope为singleton,调试一下:

这次指定scope为prototype,调试一下:

如上,当执行完创建工厂的代码时,控制台没有creat person的字样,但再往下看:

当执行完getBean方法后,出现了person无参构造这个打印语句

注:若scope为singleton,但还是想在获得对象时进行对象创建

则在bean标签中加一个lazy-init属性并且指定为true

2.初始化阶段:

Spring工厂创建完对象后,会调用初始化方法。该初始化方法是程序员提供,Spring工厂进行调用。那么如何提供初始化方法?有两种方式

方式一:InitializationBean接口

创建一个product类,实现initialization接口,并且实现其中的方法afterPropertiesSet

其实看这个方法的名字就知道,该方法的执行时期就是在property标签的set注入执行完毕后进行

下面进行文件配置:

方式二.在对象中提供普通的初始化方法,不实现接口

文件配置:

结果:

细节分析

若一个对象既实现类InitializationBean接口,又写了自己的初始化方法

那么就是先执行接口的方法,再执行自己的方法

如果还有property标签进行注入

那么就先注入,再初始化

初始化操作的意义

一般用在资源初始化中:数据库,IO,网络等

3.销毁阶段

Spring工厂在销毁之前,会调用对象的销毁方法,完成销毁操作

Spring何时销毁对象?

在调用ctx.close方法之后进行

如何销毁?

程序员定义销毁方法,Spring调用

方式一:实现DisposiableBean接口的destroy方法

文件配置:

效果:

为什么调用不了close方法呢?因为编译阶段不知道ctx的具体类型,不知道是classpathxmlapplicationcontext调用的还是webxmlapplicationcontext调用的,所以要向下转型,进行强制类型转换:

方式二:不实现接口,提供普通的销毁方法

细节分析

注意:销毁操作只适用于scope=singleton

销毁操作主要是指资源释放的操作,比如io,connection等

十一.配置文件参数化

指的是把Spring配置文件中,需要经常进行修改的字符串信息,转移到一个更小的配置文件中。规定此配置文件就是.properties文件。

有需要经常修改的字符串吗?当然有,就比如之前讲到的connection的创建,进行依赖注入,那些信息就很可能要修改(比如要换一个数据库)

1.好处

将进场需要修改的字符串转移到小的配置文件中,更利于修改,在改动的同时不会影响Spring配置文件中的其他内容

2.开发步骤

还是使用connection举例

提供小配置文件

我们创建一个db.properties文件

也是key value模式,key可以随便起名

将Spring配置文件与小配置文件进行整合

context:property-placeholder标签,表示读取小配置文件,location属性定义了小配置文件的路径。其中,classpath是Spring配置文件中特有的关键字,代表了target目录下的classes目录。

maven项目:开发时,java和resource目录是两个目录,但是编译后就合二为一了。如何证明?target目录存放的就是编译后的文件,打开就可以看到classes目录下面就有之前java和resource目录下的所有.class文件

将字符串填充到配置文件中:

"${key}"key就是小配置文件中的key

十二.自定义类型转换器

1.什么是类型转换器

在person类中,id是int类型,而Spring配置文件也是文件,它里面的内容就应该是字符串,那它怎么就把字符串赋值给了int类型呢?这是因为Spring中内置了类型转换器Converter,并且借助接口实现(为啥是接口?因为字符串要转换成不同的类型)

但是像date类型,每个地区的人的习惯性写法不同,所以Spring就可能无法将日期字符串转化成标准的Date格式(后面会讲为什么是可能)。那该怎么办?这就要我们程序员自定义类型转换器了

2.自定义类型转换器

创建一个Converter包,Person类

再写一个converter接口的实现类

Converter是一个泛型接口,见括号中的第一个表示原始类型,第二个表示要转换成什么类型

接下来我们就要去实现convert方法,这个方法的参数就是Spring配置文件中要被转换的那个字符串

最后在Spring配置文件中配置

首先要将自己写的类型转换器创建出来,然后要在Spring中进行注册。如何注册?

Spring中提供了一个类ConverterFactoryBean,它里面存放的就是各种类型转换器,是以Set集合存储的

所以要想使用自己的类型转换器,就要将自己写的类型转换器存放到set集合中,就如上所示。这样,就能将字符串转化成Date类型啦

3.细节分析

yyyy-MM-dd

它是MydateConverter的依赖项,所以可以对它进行依赖注入,也就是将它作为成员变量进行配置,后需要想修改格式,只需要修改Spring文件即可,就不用重新编译部署,就可以解耦合

为ConverterFactoryBean设标签

要注意id值必须为converterService,否则注册类型转换器不成功

Spring其实内置了日期的类型转换器

但它要求日期格式必须为:四位/二位/二位,否则无法自动转换

十三.后置处理bean

1.Spring创建对象的流程

1.反射调用构造方法

2.DI(注入)

3.InitializingBean的afterPropertySet方法对对象进行初始化

4.init-method="myInit",自定义的初始化方法

学到这里,我们就要再加一个BeanPostProcessor接口,他就是对Spring工厂所创建的对象进行再加工(就像苹果从果园中生长出来后没有直接到顾客手里,而是进行了再加工,提高了品质)

该接口中有两个方法

一个是Before,表示在调用InitializingBean接口的方法之前进行,一个是After,表示在调用了自定义的初始化方法之后进行

2.开发步骤

我们准备一个beanPost包和Category类

进行文件配置:

然后创建一个MyBeanPostProcessor类,实现BeanPostProcessor接口

我们没有实现这两个方法,但是没有报错,这是因为这两个方法是default,表示接口的默认实现

我们的目的是将name属性从小红改为小明。那该如何实现这两个方法呢?如下:

在实际开发中,很少会进行初始化操作,所以上面这两个方法实现一个即可,一般是实现after方法,另一个就返回bean。

最后进行文件配置

效果:

运行后竟然报错了!这是我们之前写的类型转换器不能转换成Category类。这是啥错误?

仔细看这个配置文件,里面有好多个bean标签,最后一个bean标签就是在创建beanPostProcessor的实现类,事实上,同一个配置文件中的对象都要调用BeanPostProcessor接口的方法,即下面我们重写的这个方法:

注意这个bean,这个bean可不是单单是Category对象,而将会是同一个配置文件中的所有对象,所以在话蓝线的这一句就会出现类型转换异常。那么该如何解决此问题?如下:

首先用instanceof判断一下是否是Category类,如果是才能进行下面的修改操作,不是就直接返回,返回的时候也是直接返回bean对象即可

所以一定要注意:​​​​​​​beanPostProcessor会对Spring文件中的所有对象进行加工,所以一定要进行类型匹配的判断。

相关推荐
湫ccc2 分钟前
Python简介以及解释器安装(保姆级教学)
开发语言·python
程序伍六七6 分钟前
day16
开发语言·c++
wkj00110 分钟前
php操作redis
开发语言·redis·php
武子康12 分钟前
大数据-230 离线数仓 - ODS层的构建 Hive处理 UDF 与 SerDe 处理 与 当前总结
java·大数据·数据仓库·hive·hadoop·sql·hdfs
武子康14 分钟前
大数据-231 离线数仓 - DWS 层、ADS 层的创建 Hive 执行脚本
java·大数据·数据仓库·hive·hadoop·mysql
极客代码15 分钟前
【Python TensorFlow】进阶指南(续篇三)
开发语言·人工智能·python·深度学习·tensorflow
苏-言21 分钟前
Spring IOC实战指南:从零到一的构建过程
java·数据库·spring
土豆湿21 分钟前
拥抱极简主义前端开发:NoCss.js 引领无 CSS 编程潮流
开发语言·javascript·css
界面开发小八哥28 分钟前
更高效的Java 23开发,IntelliJ IDEA助力全面升级
java·开发语言·ide·intellij-idea·开发工具
草莓base41 分钟前
【手写一个spring】spring源码的简单实现--容器启动
java·后端·spring