Spring中基于xml配置管理Bean的步驟
前言
在之前的學(xué)習(xí)中我們知道,容器是一個(gè)空間的概念,一般理解為可盛放物體的地方。在Spring容器通常理解為BeanFactory或者ApplicationContext。我們知道spring的IOC容器能夠幫我們創(chuàng)建對(duì)象,對(duì)象交給spring管理之后我們就不用手動(dòng)去new對(duì)象。
那么Spring是如何管理Bean的呢?
一、概念
簡(jiǎn)而言之,Spring bean是Spring框架在運(yùn)行時(shí)管理的對(duì)象。Spring bean是任何Spring應(yīng)用程序的基本構(gòu)建塊。你編寫的大多數(shù)應(yīng)用程序邏輯代碼都將放在Spring bean中。
Spring bean的管理包括:
- 創(chuàng)建一個(gè)對(duì)象
- 提供依賴項(xiàng)(例如其他bean,配置屬性)
- 攔截對(duì)象方法調(diào)用以提供額外的框架功能
- 銷毀一個(gè)對(duì)象
Spring bean是框架的基本概念。作為Spring的用戶,你應(yīng)該對(duì)這個(gè)核心抽象有深刻的理解。
二、創(chuàng)建Bean對(duì)象的三種方式
2.1、使用默認(rèn)構(gòu)造函數(shù)創(chuàng)建方式
2.1.1、定義Bean
public class UserServiceImpl { }
2.1.2、主配置文件中配置bean
<!-- 方式一:使用默認(rèn)構(gòu)造函數(shù)方式創(chuàng)建Bean --> <beans> <bean id="userService" class="cn.bdqn.UserServiceImpl"></bean> </beans>
2.1.3、測(cè)試Bean
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); // 2、從容器中根據(jù)id獲取對(duì)象(bean) UserServiceImpl userService = (UserServiceImpl) ac.getBean("userService"); // 3、打印bean System.out.println(userService); }
2.1.4、注意點(diǎn)
此種方式采用的就是通過默認(rèn)構(gòu)造函數(shù)的方式創(chuàng)建Bean,假設(shè)我們給UserServiceImpl添加了一個(gè)帶參的構(gòu)造方法,則運(yùn)行會(huì)報(bào)錯(cuò),原因在于當(dāng)我們?yōu)槟硞€(gè)類自定義構(gòu)造方法的時(shí)候,Java編譯器便不會(huì)為該類提供默認(rèn)的不帶參數(shù)的構(gòu)造方法了。
2.2、使用工廠中的實(shí)例方法創(chuàng)建方式
2.2.1、定義工廠
// UserService的工廠,作用是創(chuàng)建UserServiceBean對(duì)象 public class UserServiceImplFactory { public UserServiceImpl createUserService(){ return new UserServiceImpl(); } }
2.2.2、定義Bean
public class UserServiceImpl { }
2.2.3、主配置文件中配置Bean
<beans> <!-- 方式二:使用工廠中提供的實(shí)例方法創(chuàng)建Bean --> <!-- 第一步:把該工廠定義出來 --> <bean id="userServiceFactory" class="cn.bdqn.UserServiceImplFactory"/> <!-- 第二步:定義Bean(通過userServiceFactory中提供的實(shí)例方法)--> <bean id="userService" factory-bean="userServiceFactory" factory-method="createUserService"/> </beans>
2.2.4、測(cè)試
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); // 2、從容器中根據(jù)id獲取對(duì)象(bean) UserServiceImpl userService = (UserServiceImpl) ac.getBean("userService"); // 3、打印bean System.out.println(userService); }
2.3、使用工廠中的靜態(tài)方法創(chuàng)建方式
2.3.1、定義工廠
// UserService的工廠,作用是創(chuàng)建UserServiceBean對(duì)象 public class UserServiceImplFactory { public static UserServiceImpl createUserService(){ return new UserServiceImpl(); } }
2.3.2、定義Bean
public class UserServiceImpl { }
2.3.3、主配置文件中配置Bean
<beans> <!-- 方式三:使用工廠中提供的靜態(tài)方法創(chuàng)建Bean --> <!-- 定義Bean(通過工廠類的靜態(tài)方法創(chuàng)建) --> <bean id="userService" class="cn.bdqn.UserServiceImplFactory" factory-method="createUserService"> </bean> </beans>
2.3.4、測(cè)試
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); // 2、從容器中根據(jù)id獲取對(duì)象(bean) UserServiceImpl userService = (UserServiceImpl) ac.getBean("userService"); // 3、打印bean System.out.println(userService); }
三、Bean對(duì)象的作用域
3.1、說明
? Spring對(duì)Bean的默認(rèn)的作用域(作用范圍)是singleton【單例】
3.2、作用域類型
- singleton:?jiǎn)卫模J(rèn)值),只會(huì)new一次。
- prototype:多例的,用到一次就會(huì)new一次。
- request:作用于web應(yīng)用的請(qǐng)求范圍,Spring創(chuàng)建這個(gè)類之后,將這個(gè)類存到request范圍內(nèi)。
- session:應(yīng)用于web項(xiàng)目的會(huì)話范圍,Spring創(chuàng)建這個(gè)類之后,將這個(gè)類存到session范圍內(nèi)。
- global-session:作用于集群環(huán)境的會(huì)話范圍(全局會(huì)話范圍),當(dāng)不是集群環(huán)境時(shí),它就是session。
3.3、注意細(xì)節(jié)
實(shí)際開發(fā)中用得最多的就是singleton和prototype,在整合struts2的時(shí)候使用prototype,在整合SpringMVC的時(shí)候使用singleton。
3.4、如何修改Bean的作用域
bean標(biāo)簽的scope屬性,作用:指定bean的作用范圍。
3.5、測(cè)試
3.5.1、測(cè)試singleton單例
public class UserServiceImpl { }
public class UserServiceImpl { }
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); // 2、從容器中根據(jù)id獲取對(duì)象(bean) UserServiceImpl userService1 = (UserServiceImpl) ac.getBean("userService"); UserServiceImpl userService2 = (UserServiceImpl) ac.getBean("userService"); // 3、打印bean System.out.println(userService1 == userService2); // true }
3.5.2、測(cè)試prototype多例
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); // 2、從容器中根據(jù)id獲取對(duì)象(bean) UserServiceImpl userService1 = (UserServiceImpl) ac.getBean("userService"); UserServiceImpl userService2 = (UserServiceImpl) ac.getBean("userService"); // 3、打印bean System.out.println(userService1 == userService2); // true }
<bean id="userService" class="cn.bdqn.UserServiceImpl" scope="prototype"/>
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); // 2、從容器中根據(jù)id獲取對(duì)象(bean) UserServiceImpl userService1 = (UserServiceImpl) ac.getBean("userService"); UserServiceImpl userService2 = (UserServiceImpl) ac.getBean("userService"); // 3、打印bean System.out.println(userService1 == userService2); // false }
四、Bean對(duì)象的生命周期
4.1、單例對(duì)象
4.1.1、說明
出生:當(dāng)容器創(chuàng)建時(shí)對(duì)象出生
活著:只要容器還在,對(duì)象一直活著
死亡:容器銷毀,對(duì)象消亡
4.1.2、測(cè)試
4.1.2.1、定義Bean
public class UserServiceImpl { public UserServiceImpl(){ System.out.println("對(duì)象的構(gòu)造方法執(zhí)行了"); } public void init(){ System.out.println("對(duì)象初始化了"); } public void destroy(){ System.out.println("對(duì)象銷毀了"); } }
4.1.2.2、主配置文件中配置Bean
<beans> <bean id="userService" class="cn.bdqn.UserServiceImpl" scope="singleton" init-method="init" destroy-method="destroy"/> </beans>
4.1.2.3、測(cè)試
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); ac.close(); } // 結(jié)果:對(duì)于單例對(duì)象來說,只要容器創(chuàng)建了,那么對(duì)象就創(chuàng)建了。類似于立即加載。
4.1.2.4、測(cè)試結(jié)果
對(duì)象的構(gòu)造方法執(zhí)行了
對(duì)象初始化了
對(duì)象銷毀了
總結(jié):?jiǎn)卫龑?duì)象的生命周期和容器相同
4.2、多例對(duì)象
4.2.1、說明
出生:當(dāng)我們使用對(duì)象時(shí)spring框架為我們創(chuàng)建
活著:對(duì)象只要是在使用過程中就一直活著。
死亡:當(dāng)對(duì)象長(zhǎng)時(shí)間不用,且沒有別的對(duì)象引用時(shí),由Java的垃圾回收器回收
4.2.2、測(cè)試
4.2.2.1、定義Bean
public class UserServiceImpl { public UserServiceImpl(){ System.out.println("對(duì)象的構(gòu)造方法執(zhí)行了"); } public void init(){ System.out.println("對(duì)象初始化了"); } public void destroy(){ System.out.println("對(duì)象銷毀了"); } }
4.2.2.2、主配置文件中配置Bean
<beans> <bean id="userService" class="cn.bdqn.UserServiceImpl" scope="prototype" init-method="init" destroy-method="destroy"/> </beans>
4.2.2.3、測(cè)試1
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); ac.close(); } // 結(jié)果:什么都不輸出,說明容器啟動(dòng)的時(shí)候,對(duì)于多例對(duì)象來說并不會(huì)創(chuàng)建
4.2.2.4、測(cè)試2
@Test public void testUserService() throws Exception{ // 1、讀取主配置文件信息,獲取核心容器對(duì)象 ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); UserServiceImpl userService = (UserServiceImpl) ac.getBean("userService"); System.out.println(userService); ac.close(); } /** 結(jié)果: 對(duì)象的構(gòu)造方法執(zhí)行了 對(duì)象初始化了 說明: 對(duì)于多例對(duì)象來說,只有等到真正使用到該對(duì)象的時(shí)候才會(huì)創(chuàng)建。類似于懶加載。 **/
? 對(duì)于多例的Bean,Spring框架是不負(fù)責(zé)管理的
五、總結(jié)
到此這篇關(guān)于Spring中基于xml配置管理Bean的步驟的文章就介紹到這了,更多相關(guān)spring 管理Bean內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)簡(jiǎn)單斗地主(看牌排序)
這篇文章主要介紹了java實(shí)現(xiàn)簡(jiǎn)單斗地主,看牌進(jìn)行排序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2010-11-11Spring Security OAuth 自定義授權(quán)方式實(shí)現(xiàn)手機(jī)驗(yàn)證碼
這篇文章主要介紹了Spring Security OAuth 自定義授權(quán)方式實(shí)現(xiàn)手機(jī)驗(yàn)證碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02使用@PathVariable注解如何實(shí)現(xiàn)動(dòng)態(tài)傳值
這篇文章主要介紹了使用@PathVariable注解如何實(shí)現(xiàn)動(dòng)態(tài)傳值,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10Java并發(fā)編程包中atomic的實(shí)現(xiàn)原理示例詳解
這篇文章主要給大家介紹了關(guān)于Java并發(fā)編程包中atomic的實(shí)現(xiàn)原理,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-09-09java 實(shí)現(xiàn)隨機(jī)數(shù)組輸出及求和實(shí)例詳解
這篇文章主要介紹了java 實(shí)現(xiàn)隨機(jī)數(shù)組輸出及求和實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2016-11-11Java代碼如何判斷l(xiāng)inux系統(tǒng)windows系統(tǒng)
這篇文章主要介紹了Java代碼如何判斷l(xiāng)inux系統(tǒng)windows系統(tǒng)問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01Java Volatile應(yīng)用單例模式實(shí)現(xiàn)過程解析
這篇文章主要介紹了Java Volatile應(yīng)用單例模式實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11IntelliJ IDEA 無法正常使用SVN的問題和完美解決辦法
這篇文章主要介紹了IntelliJ IDEA 無法正常使用SVN的問題和解決辦法,本文給大家分享完美解決方案,通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-08-08