簡(jiǎn)單了解Spring IoC相關(guān)概念原理
Spring Ioc是Spring框架的基礎(chǔ),本文會(huì)簡(jiǎn)單的介紹下Spring Ioc。
Sprong Ioc即控制反轉(zhuǎn),它是一種通過(guò)描述(在java中可以是XML或注解)并通過(guò)第三方去產(chǎn)生或獲取特定對(duì)象的方式。
Spring IoC容器
1、Spring IoC容器的設(shè)計(jì)
Spring IoC容器的設(shè)計(jì)主要是基于BeanFactory和ApplicationContext這兩個(gè)接口,其中ApplicationContext是BeanFactory的一個(gè)子接口。也就是說(shuō),BeanFactory是Spring IoC容器定義的最底層接口,而ApplicationContext是其高級(jí)接口之一,因此大部分情況下會(huì)使用后者作為Spring IoC容器。
1.1 ClassPathXmlAppLicationContext
首先我們來(lái)認(rèn)識(shí)一下ApplicationContext的子類ClassPathXmlAppLicationContext。先創(chuàng)建一個(gè).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
http://www.springframework.org/schema/beans/spring-beans-4.0.xsd">
<bean id="source" class="com.ssm.chapter.pojo.Source">
<property name="fruit" value="橙汁" />
<property name="sugar" value="少糖" />
<property name="size" value="大杯" />
</bean>
<bean id="juiceMaker" class="com.ssm.chapter.pojo.JuiceMaker" >
<property name="beverageShop" value="貢茶" />
<property name="source" ref="source" />
</bean>
</beans>
這里定義了兩個(gè)bean,這樣Spring IoC容器在初始化的時(shí)候就可以找到它們,然后使用ClassPathXmlAppLicationContext容器就可以將其初始化,代碼清單如下:
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-cfg.xml");
JuiceMaker juiceMaker = (JuiceMaker) ctx.getBean("juiceMaker");
System.out.println(juiceMaker.makeJuice());
這樣就會(huì)使用Application的實(shí)現(xiàn)類ClassPathXmlAppLicationContext去初始化Spring IoC,然后開(kāi)發(fā)者就可以通過(guò)Ioc容器獲取資源了。
1.2 Spring Bean的生命周期
Spring IoC容器的本質(zhì)就是為了管理Bean。生命周期主要是為了了解Spring IoC容器初始化和銷毀Bean的過(guò)程,通過(guò)對(duì)它的學(xué)習(xí)就可以知道如何在初始和銷毀的時(shí)候加入自定義的方法,以滿足特定的需求。注:Spring IoC容器初始化和銷毀Bean的過(guò)程我這里就不介紹了啊,在網(wǎng)上很容易找到,這里主要是通過(guò)代碼去實(shí)現(xiàn)生命周期的過(guò)程。
除了了解生命周期的步驟之外,還要知道生命周期的接口是針對(duì)設(shè)么而言的,首先介紹生命周期的步驟:
①如果Bean實(shí)現(xiàn)了接口BeanNameAware,那么就會(huì)調(diào)用setBeanName方法。
②如果Bean實(shí)現(xiàn)了接口BeanFactoryAware,那么就會(huì)調(diào)用setBeanFactory方法。
③如果Bean實(shí)現(xiàn)了接口ApplicationContextAware,且Spring IoC容器也是ApplicationContext的一個(gè)實(shí)現(xiàn)類,那么就會(huì)調(diào)用setApplicationContext方法。
④如果Bean實(shí)現(xiàn)了接口BeanPostProcessor的,那么就會(huì)調(diào)用postProcessBeforeInitialization方法。
⑤如果Bean實(shí)現(xiàn)了接口BeanFactoryPostProcess,那么就會(huì)調(diào)用afterPropertiesSet方法。
⑥如果Bean自定義了初始化方法,它就會(huì)地用用已定義的初始化方法。
⑦如果Bean實(shí)現(xiàn)了接口BeanPostProcessor,那么就會(huì)調(diào)用postProcessAfterInitialization方法,之后這個(gè)bean就會(huì)完成了初始化,開(kāi)發(fā)者就可以從Spring IoC中獲取Bean的服務(wù)。
⑧如果Bean實(shí)現(xiàn)了接口DisposableBean,那么就會(huì)調(diào)用destroy的方法。
⑨如果定義了自定義銷毀方法,那么就會(huì)調(diào)用它。
此外,上面大部分的接口是針對(duì)單個(gè)Bean而言的;而B(niǎo)eanPostProcessor接口則是針對(duì)所有Bean而言的。為了測(cè)試BeanPostProcessor接口,可以寫(xiě)一個(gè)實(shí)現(xiàn)類:
package com.ssm.chapter.bean;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
public class BeanPostProcessorImpl implements BeanPostProcessor {
@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
System.out.println("[" + bean.getClass().getSimpleName() + "]對(duì)象" + beanName + "開(kāi)始初始化");
return bean;
}
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
System.out.println("[" + bean.getClass().getSimpleName() + "]對(duì)象" + beanName + "實(shí)例化完成");
return bean;
}
}
這樣BeanPostProcessor就被我們用代碼實(shí)現(xiàn)了,他會(huì)處理Spring IoC容器中的所有Bean。
為了更好的展示生命周期的內(nèi)容,將上面的代碼中JuiceMaker類進(jìn)行修改:
package com.ssm.chapter.pojo;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
import org.springframework.beans.factory.BeanNameAware;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
public class JuiceMaker implements BeanNameAware, BeanFactoryAware, ApplicationContextAware,
InitializingBean, DisposableBean{
private String beverageShop = null;
private Source source = null;
public String getBeverageShop() {
return beverageShop;
}
public void setBeverageShop(String beverageShop) {
this.beverageShop = beverageShop;
}
public Source getSource() {
return source;
}
public void setSource(Source source) {
this.source = source;
}
public void init() {
System.out.println("[" + this.getClass().getSimpleName() + "]執(zhí)行自定義初始化方法");
}
public void myDestroy() {
System.out.println("[" + this.getClass().getSimpleName() + "]執(zhí)行自定義銷毀方法");
}
public String makeJuice() {
String juice = "這是一杯由" + beverageShop + "飲品店,提供的" + source.getSize() +source.getSugar() +
source.getFruit();
return juice;
}
@Override
public void setBeanName(String name) {
System.out.println("[" + this.getClass().getSimpleName() + "]調(diào)用BeanNameAware接口的setBeanName方法");
}
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
System.out.println("[" + this.getClass().getSimpleName() + "]調(diào)用BeanFactoryAware接口的setBeanFactory方法");
}
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
System.out.println("[" + this.getClass().getSimpleName() + "]調(diào)用ApplicationContextAware接口的setApplicationContext方法");
}
@Override
public void destroy() throws Exception {
System.out.println("[" + this.getClass().getSimpleName() + "]調(diào)用DisposableBean接口的destroy方法");
}
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("[" + this.getClass().getSimpleName() + "]調(diào)用InitializingBean接口的afterPropertiesSet方法");
}
}
這個(gè)類實(shí)現(xiàn)了所以生命周期中的方法,以便以觀察生命周期,其中init方法是自定義的初始化方法,而myDestroy方法是自定義的銷毀方法,為了進(jìn)一步使用這兩個(gè)自定義方法,在描述Bean的時(shí)候,也要在.xml中進(jìn)行如下聲明:
<bean id="beanPostProcessor"
class="com.ssm.chapter.bean.BeanPostProcessorImpl" />
<bean id="source" class="com.ssm.chapter.pojo.Source">
<property name="fruit" value="橙汁" />
<property name="sugar" value="少糖" />
<property name="size" value="大杯" />
</bean>
<bean id="juiceMaker" class="com.ssm.chapter.pojo.JuiceMaker" init-method="init" destroy-method="myDestroy">
<property name="beverageShop" value="貢茶" />
<property name="source" ref="source" />
</bean>
這里定義了id為JuiceMaker的Bean,其屬性init-menth就是自定義的初始化方法,而destroy-method為自定義的銷毀方法。下面是測(cè)試代碼清單:
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("spring-cfg.xml");
JuiceMaker juiceMaker = (JuiceMaker) ctx.getBean("juiceMaker");
System.out.println(juiceMaker.makeJuice());
ctx.close();
日志如下:
[Source]對(duì)象source開(kāi)始初始化
[Source]對(duì)象source實(shí)例化完成
[JuiceMaker]調(diào)用BeanNameAware接口的setBeanName方法
[JuiceMaker]調(diào)用BeanFactoryAware接口的setBeanFactory方法
[JuiceMaker]調(diào)用ApplicationContextAware接口的setApplicationContext方法
[JuiceMaker]對(duì)象juiceMaker開(kāi)始初始化
[JuiceMaker]調(diào)用InitializingBean接口的afterPropertiesSet方法
[JuiceMaker]執(zhí)行自定義初始化方法
[JuiceMaker]對(duì)象juiceMaker實(shí)例化完成
這是一杯由貢茶飲品店,提供的大杯少糖橙汁
[JuiceMaker]調(diào)用DisposableBean接口的destroy方法
[JuiceMaker]執(zhí)行自定義銷毀方法
從日志中可以看出,生命周期中的方法都被執(zhí)行了。也可以看到BeanPostProcessor針對(duì)的是全部Bean。我們也可以自定義初始化和銷毀Bean的方法。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java如何實(shí)現(xiàn)壓縮文件與解壓縮zip文件
這篇文章主要介紹了Java如何實(shí)現(xiàn)壓縮文件與解壓縮zip文件問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12
Vert-x-通過(guò)異步的方式使用JDBC連接SQL
在這篇文章中,我們將會(huì)看到怎樣在vert.x應(yīng)用中使用HSQL,當(dāng)然也可以使用任意JDBC,以及使用vertx-jdbc-client提供的異步的API,這篇文章的代碼在github2016-01-01
SpringCloudStream中的消息分區(qū)數(shù)詳解
這篇文章主要介紹了SpringCloudStream中的消息分區(qū)數(shù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12
啟用springboot security后登錄web頁(yè)面需要用戶名和密碼的解決方法
這篇文章主要介紹了啟用springboot security后登錄web頁(yè)面需要用戶名和密碼的解決方法,也就是使用默認(rèn)用戶和密碼登錄的操作方法,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-02-02
Spring Task定時(shí)任務(wù)每天零點(diǎn)執(zhí)行一次的操作
這篇文章主要介紹了Spring Task定時(shí)任務(wù)每天零點(diǎn)執(zhí)行一次的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
使用lombok注解導(dǎo)致mybatis-plus TypeHandler失效的解決
這篇文章主要介紹了使用lombok注解導(dǎo)致mybatis-plus TypeHandler失效的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07
關(guān)于Jmeter接口測(cè)試實(shí)戰(zhàn)-Cookies
這篇文章主要介紹了關(guān)于Jmeter接口測(cè)試實(shí)戰(zhàn)-Cookies問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-03-03
ssm項(xiàng)目實(shí)現(xiàn)用戶登陸持久化(token)
這篇文章主要介紹了ssm項(xiàng)目實(shí)現(xiàn)用戶登陸持久化(token),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04

