**延遲初始化Bean:**
也叫惰性初始化,指的是不提前初始化Bean,而是隻有在真正使用的時候才創建以 及初始化Bean。
配置方式很簡單,只需要在<bean>標籤上指定"lazy-init"屬性值爲true即可延遲初始 化Bean。
Spring容器會在創建容器時提前初始化Singleton作用域的bean,Singleton就是單例 的意思,即整個容器每個bean只有一個實例。Spring容器預先初始化Bean同城能夠幫助我們提前發現配置錯誤,所以沒有什麼情況建議開啓,除非有某個bean可能需要加載很大資源,而且很有可能在整個應用程序生命週期中很有可能使用不到,可以設置爲延遲初始化。
延遲初始化的Bean通常會再第一次使用的時候被初始化,或者在被非延遲初始化Bean作爲以來對象注入時爲依賴對象注入時在會隨着初始化該Bean時初始化,因爲在這個時候使用延遲初始化Bean。
<bean id="user"
Class="org.liang.entity.User"
lazy-init="true" />
depends-on : 是指指定Bean初始化以及銷燬時的順序,使用depends-on屬性指定的Bean要初始化完畢以後纔會初始化當前的Bean,由於只有“Singleton”Bean能夠被Spring管理銷燬,所以當指定的Bean都是Singleton時,使用depends-on時,使用depends-on屬性指定的Bean要在指定的Bean之後銷燬。
ResourceBean.java
package org.liang.entity;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
/**
* Created by rcc on 2018/1/29.
*/
public class ResourceBean {
private FileOutputStream fos;
private String fileAddress;
//初始化方法
public void init(){
System.out.println("ResourceBean: ================初始化");
System.out.println("ResourceBean: ============加載");
try {
this.fos = new FileOutputStream(new File(fileAddress));
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
//銷燬資源方法
public void destroy(){
System.out.println("ResourceBean:===============銷燬");
System.out.println("ResourceBean:===============釋放資源,執行一些操作");
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
public FileOutputStream getFos() {
return fos;
}
public void setFos(FileOutputStream fos) {
this.fos = fos;
}
public String getFileAddress() {
return fileAddress;
}
public void setFileAddress(String fileAddress) {
this.fileAddress = fileAddress;
}
}
DependentBean.java
“`
package org.liang.factory;
import org.liang.entity.ResourceBean;
import java.io.IOException;
/**
* Created by rcc on 2018/1/29.
*/
public class DependentBean {
private ResourceBean resourceBean;
//寫入內容
public void write(String ss) throws IOException {
System.out.println("DependentBean:==============寫資源");
resourceBean.getFos().write(ss.getBytes());
}
//初始化方法
public void init() throws IOException {
System.out.println("DependentBean:==============初始化");
resourceBean.getFos().write("DependentBean:================初始化=======".getBytes());
}
//銷燬方法
public void destory() throws IOException {
System.out.println("DependentBean:==============銷燬");
//在銷燬之前需要向文本中寫入銷燬內容
resourceBean.getFos().write("DependentBean:==============銷燬======".getBytes());
}
public ResourceBean getResourceBean() {
return resourceBean;
}
public void setResourceBean(ResourceBean resourceBean) {
this.resourceBean = resourceBean;
}
}
“`Spring-config.xml
<!--測試 懶加載-->
<bean id="resourceBean"
class="org.liang.entity.ResourceBean"
init-method="init" destroy-method="destroy">
<property name="fileAddress" value="E:/lazy.txt" />
</bean>
<!--
init-method="init" :指定初始化方法,在構造器注入和setter注入完畢後執行。
destroy-method="destroy":指定銷燬方法,只有“singleton”作用域能銷燬,“prototype”作用域的一定不能,
其他作用域不一定能;
-->
<bean id="dependentBean"
class="org.liang.factory.DependentBean"
init-method="init"
destroy-method="destory"
depends-on="resourceBean"
autowire="byName"> <!--autowire="default"-->
<property name="resourceBean" ref="resourceBean"/>
</bean>
<!--在此配置中,resourceBean初始化在dependentBean之前被初始化,
resourceBean銷燬會在dependentBean銷燬之後執行-->
<!--
自動裝配:
default:表示使用默認的自動裝配,默認的是自動裝配需要在<beans>標籤中使用default-autowire屬性指定,其支持“no”、“byName ”、“byType”、“constructor”四種自動裝配
no:意思是不支持自動裝配,必須明確指定依賴。
byName:通過設置Bean定義屬性autowire="byName",意思是根據名字進行自動裝配,只能用於setter注入。
-->
MoreDependencyInjectTest .java
package org.liang.test;
import org.liang.factory.DependentBean;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import java.io.IOException;
/**
* Created by rcc on 2018/1/30.
*/
public class MoreDependencyInjectTest {
public static void main(String [] args){
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
DependentBean dependentBean = context.getBean(DependentBean.class);
//註冊銷燬回調否則銷燬的方法不會執行
context.registerShutdownHook();
try {
dependentBean.write("hello world!!");
} catch (IOException e) {
e.printStackTrace();
}
/*
返回的結果:
ResourceBean: ================初始化
ResourceBean: ============加載
DependentBean:==============初始化
DependentBean:==============寫資源
DependentBean:==============銷燬
ResourceBean:===============銷燬
ResourceBean:===============釋放資源,執行一些操作
*/
}
}
自動裝配
指的是由Spring來自主地注入依賴對象,無需人工參與。
目前Spring3.0支持“no”、“byName ”、“byType”、“constructor”四種自動裝配,默認是“no”指不支持自動裝配的,其中Spring3.0已不推薦使用之前版本的“autodetect”自動裝配,推薦使用Java 5+支持的(@Autowired)註解方式代替;如果想支持“autodetect”自動裝配,請將schema改爲“spring-beans-2.5.xsd”或去掉。
自動裝配的好處是減少構造器注入和setter注入配置,減少配置文件的長度。自動裝配通過配置<bean>標籤的“autowire”屬性來改變自動裝配方式。接下來讓我們挨着看下配置的含義。
一、default:表示使用默認的自動裝配,默認的自動裝配需要在<beans>標籤中使用default-autowire屬性指定,其支持“no”、“byName ”、“byType”、“constructor”四種自動裝配,如果需要覆蓋默認自動裝配,請繼續往下看;
二、no:意思是不支持自動裝配,必須明確指定依賴。
三、byName:通過設置Bean定義屬性autowire="byName",意思是根據名字進行自動裝配,只能用於setter注入。比如我們有方法“setHelloApi”,則“byName”方式Spring容器將查找名字爲helloApi的Bean並注入,如果找不到指定的Bean,將什麼也不注入。
四、“byType”:通過設置Bean定義屬性autowire="byType",意思是指根據類型注入,用於setter注入,比如如果指定自動裝配方式爲“byType”,而“setHelloApi”方法需要注入HelloApi類型數據,則Spring容器將查找HelloApi類型數據,如果找到一個則注入該Bean,如果找不到將什麼也不注入,如果找到多個Bean將優先注入<bean>標籤“primary”屬性爲true的Bean,否則拋出異常來表明有個多個Bean發現但不知道使用哪個
五、“constructor”:通過設置Bean定義屬性autowire="constructor",功能和“byType”功能一樣,根據類型注入構造器參數,只是用於構造器注入方式,