什么是IOC容器叠穆?
IOC即反轉(zhuǎn)控制瘦材,創(chuàng)建對象的權(quán)利交給容器來完成期奔,而程序要做的僅僅是使用對象侧馅。
Spring 容器是 Spring 框架的核心。容器將創(chuàng)建對象呐萌,把它們連接在一起馁痴,配置它們,并管理他們的整個生命周期從創(chuàng)建到銷毀肺孤。Spring 容器使用依賴注入(DI)來管理組成一個應(yīng)用程序的組件罗晕。這些對象被稱為 Spring Beans
Spring中的IOC容器的使用
如何在IOC容器中配置bean?
使用標簽來將一個類配置到IOC容器中
<!-- bean標簽:用來向IOC容器配置一個bean
id:用來唯一標識這個bean對象
class:指明該bean對象的全類名
property標簽:向bean中注入屬性
-->
<bean id="student" class="com.cn.cmc.beans.Student" >
<property name="name" value="葉清逸"></property>
</bean>
如何獲取IOC容器赠堵?
Spring提供兩種IOC容器:
ApplicationContext容器
BeanFactory容器(已廢棄)
獲取ApplicationContext容器
/**
* ApplicationContext代表IOC容器
* ClassPathXmlApplicationContext從類路徑下讀取配置文件
* FileSystemXmlApplicationContext從文件系統(tǒng)路徑下讀取配置文件
* WebXmlApplicationContext:在web 應(yīng)用程序的范圍內(nèi)加載在 XML 文件中已被定義的 bean小渊。
*/
ApplicationContext context = new FileSystemXmlApplicationContext("config/beans.xml") ;
獲取BeanFactory容器(已廢棄)
/**
- XmlBeanFactory生成工廠bean
- ClassPathResource加載配置文件
*/
XmlBeanFactory factory = new XmlBeanFactory(new ClassPathResource("config/beans.xml")) ;
bean管理
從IOC容器中獲取bean
方式一:通過id獲取bean
/方式一:通過id獲取bean/
Student student = context.getBean(Student.class);
System.out.println(student.getName());
方式二:通過.class獲取bean
/方式二:通過.class文件獲取bean/
Student s = (Student)context.getBean("student") ;
System.out.println(s.getName());
方式三:通過工廠方法來獲取bean
靜態(tài)工廠方法
方法類
package com.cn.cmc.beans;
import java.util.HashMap;
import java.util.Map;
public class StaticStudentFactory {
private static Map<String,Student> map ;
static {
map = new HashMap<String, Student>();
map.put("葉清逸", new Student(100001,"葉清逸",'M',25)) ;
map.put("張三", new Student(100002,"張三",'F',22)) ;
}
/*通過靜態(tài)方法獲取bean的實例*/
static Student getBean(String name){
return map.get(name) ;
}
}
xml中的配置:
<!-- 通過靜態(tài)工廠方法來獲取實例
factory-method:對應(yīng)工廠中配置的獲取bean的方法名
constructor-arg:對用獲取bean方法傳入的參數(shù)
-->
<bean id="student1" class="com.cn.cmc.beans.StaticStudentFactory" factory-method="getBean">
<constructor-arg value="葉清逸"></constructor-arg>
</bean>
bean的屬性注入
方式一:屬性注入(通過setter方法)
<property name="name" value="葉清逸"></property>
方式二:構(gòu)造方法注入
<bean id="student" class="com.cn.cmc.beans.Student" >
<!-- constructor-arg標簽:使用構(gòu)造方法來傳入?yún)?shù)
name:對應(yīng)傳入屬性的名
value:對應(yīng)傳入屬性的值
index:對應(yīng)傳入屬性在類中的位置
type:對應(yīng)傳入屬性在類中的類型
ref:bean之間的引用
注:index和type主要是為了解決傳入時產(chǎn)生歧義的問題,可以不寫
在類中要有對應(yīng)的構(gòu)造方法茫叭,可以沒有setter方法
-->
<constructor-arg name="id" value="100001"></constructor-arg>
<constructor-arg name="name">
<value>葉清逸</value>
</constructor-arg>
<constructor-arg name="sex" value="M" index="2" type="char"></constructor-arg>
<constructor-arg name="age" value="25"></constructor-arg>
</bean>
注:若傳入值有特殊字符酬屉,可以使用<[CDATA[]]包裹起來>
注入的一些其他細節(jié)
使用ref屬性引用其他bean
<bean id="student" class="com.cn.cmc.beans.Student" >
<constructor-arg name="id" value="100001"></constructor-arg>
<constructor-arg name="name">
<value>葉清逸</value>
</constructor-arg>
<constructor-arg name="sex" value="M" index="2" type="char"></constructor-arg>
<constructor-arg name="age" value="25"></constructor-arg>
<!-- ref:引用其他已配置的bean -->
<constructor-arg name="phone" ref="phone"></constructor-arg>
</bean>
<bean id="phone" class="com.cn.cmc.beans.Phone">
<property name="id" value="200001"></property>
<property name="name" value="蘋果手機"></property>
</bean>
屬性內(nèi)配置內(nèi)部bean
<bean id="student" class="com.cn.cmc.beans.Student" >
<constructor-arg name="id" value="100001"></constructor-arg>
<constructor-arg name="name">
<value>葉清逸</value>
</constructor-arg>
<constructor-arg name="sex" value="M" index="2" type="char"></constructor-arg>
<constructor-arg name="age" value="25"></constructor-arg>
<!-- ref:引用其他已配置的bean -->
<constructor-arg name="phone">
<bean class="com.cn.cmc.beans.Phone">
<property name="id" value="200001"></property>
<property name="name" value="蘋果手機"></property>
</bean>
</constructor-arg>
</bean>
注:內(nèi)部 bean不能被外部引用
配置級聯(lián)屬性
<bean id="student" class="com.cn.cmc.beans.Student" >
<constructor-arg name="id" value="100001"></constructor-arg>
<constructor-arg name="name">
<value>葉清逸</value>
</constructor-arg>
<constructor-arg name="sex" value="M" index="2" type="char"></constructor-arg>
<constructor-arg name="age" value="25"></constructor-arg>
<!-- ref:引用其他已配置的bean -->
<constructor-arg name="phone" ref="phone"></constructor-arg>
<!-- 配置級聯(lián)屬性(要有對應(yīng)的bean和setter方法) -->
<property name="phone.name" value="小米手機"></property>
</bean>
<bean id="phone" class="com.cn.cmc.beans.Phone">
<property name="id" value="200001"></property>
<property name="name" value="蘋果手機"></property>
</bean>
Collection的注入(set,map,list,property)
<property name="phoneSet">
<!-- 使用<set>標簽來注入set -->
<set>
<!-- ref標簽:表示關(guān)聯(lián)的bean
bean:已配置的bean的id -->
<ref bean="phone1"/>
<ref bean="phone2"/>
<!-- 也可以注入內(nèi)部bean -->
<bean class="com.cn.cmc.beans.Phone">
<property name="id" value="200003"></property>
<property name="name" value="華為手機"></property>
</bean>
</set>
</property>
<property name="phoneList">
<!-- 使用<list>標簽來注入list -->
<list>
<!-- ref標簽:表示關(guān)聯(lián)的bean
bean:已配置的bean的id
-->
<ref bean="phone1"/>
<ref bean="phone2"/>
<!-- 也可以注入內(nèi)部bean -->
<bean class="com.cn.cmc.beans.Phone">
<property name="id" value="200003"></property>
<property name="name" value="華為手機"></property>
</bean>
<ref bean="phone1"/>
<ref bean="phone2"/>
</list>
</property>
<property name="phoneMap">
<!-- 使用<map>標簽來注入map -->
<map>
<!-- entry標簽:表示map中的元素
key:注入的鍵
value:注入的值
key-ref:注入鍵的關(guān)聯(lián)bean的id
value-ref:祝福值關(guān)聯(lián)bean的id
-->
<entry key="1" value-ref="phone1"></entry>
<entry key="2" value-ref="phone2"></entry>
<!-- 也可以使用注入內(nèi)部類的方式 -->
<entry key="3">
<bean class="com.cn.cmc.beans.Phone">
<property name="id" value="200003"></property>
<property name="name" value="華為手機"></property>
</bean>
</entry>
</map>
</property>
<property name="phoneProp">
<!-- 使用prop配置property屬性 -->
<props>
<prop key="1">cmc</prop>
<prop key="2">root</prop>
</props>
</property>
使用util標簽配置共享的bean
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util-2.0.xsd">
<!-- 使用util標簽來配置共享的bean -->
<util:list id="phonelist">
<ref bean="phone1"/>
<ref bean="phone2"/>
</util:list>
使用p命名空間配置bean
<bean id="student2" class="com.cn.cmc.beans.Student" p:id="100002" p:name="張三"
p:age="22" p:sex="F"></bean>
bean的自動裝配
IOC容器可以自動裝配bean,我們要做的僅僅是在bean的autowire屬性中配置裝配方式:
byType 按類型自動裝配
byName 按名稱自動裝配
byConstructor 按構(gòu)造器自動裝配(不推薦)
byName 按名稱自動裝配bean
<bean id="student2" class="com.cn.cmc.beans.Student" p:id="100002" p:name="張三"
p:age="22" p:sex="F" autowire="byName"></bean>
注:通過byName自動裝配杂靶,屬性名必須和bean中定義的屬性名一致梆惯,否則裝配為空
byType 按類型自動裝配bean
<bean id="student3" class="com.cn.cmc.beans.Student" p:id="100002" p:name="張三"
p:age="22" p:sex="F" autowire="byType"></bean>
注:通過byType自動裝配酱鸭,已配置的 bean中只有一個該類型的 bean時會裝配成功,有多個時會報錯
bean之間的關(guān)系(繼承垛吗,依賴)
bean之間的關(guān)系有:
繼承:子bean會繼承父bean中的屬性
依賴:實例化bean時要配置依賴的bean凹髓,否則無法實例化該bean
繼承與依賴的例子
<bean id="phone" class="com.cn.cmc.beans.Phone"
p:id="200001" p:name="蘋果手機"></bean>
<bean id="student" class="com.cn.cmc.beans.Student"
p:id="100003" p:name="李四" p:age="24" p:sex="M" abstract="true"></bean>
<bean id="student1" class="com.cn.cmc.beans.Student"
parent="student" p:name="王五" p:id="100004" p:phone-ref="phone"
depends-on="phone"></bean>
bean的作用域配置
Spring中可以使用scope屬性來配置bean的作用域:
singleton:單例,在初始化配置文件時生成單例bean對象
prototype:原型的怯屉,在初始化配置文件時不生成bean對象蔚舀,使用時返回不同的bean對象
request:web環(huán)境下每一個request請求都會返回一個不同的bean,只在本次請求中有效
session:web環(huán)境下每一個request請求都會返回一個不同的bean锨络,在session中有效
<bean id="student" class="com.cn.cmc.beans.Student"
p:id="100001" p:name="葉清逸" p:age="25" p:sex="M" scope="singleton"></bean>
IOC容器中bean的生命周期
通過init-method和destroy-method
先看例子
<bean id="student" class="com.cn.cmc.beans.Student"
p:id="100001" p:name="葉清逸" p:age="25" p:sex="M" scope="singleton"
init-method="init" destroy-method="destroy"></bean>
執(zhí)行結(jié)果:
七月 08, 2018 4:21:13 下午 org.springframework.context.support.FileSystemXmlApplicationContext prepareRefresh
信息: Refreshing org.springframework.context.support.FileSystemXmlApplicationContext@11028347: startup date [Sun Jul 08 16:21:13 CST 2018]; root of context hierarchy
七月 08, 2018 4:21:13 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from file [C:\Users\Mr_Chuan\workspace\HelloSpring\config\bean-cycle.xml]
bean初始化...
七月 08, 2018 4:21:14 下午 org.springframework.context.support.FileSystemXmlApplicationContext doClose
信息: Closing org.springframework.context.support.FileSystemXmlApplicationContext@11028347: startup date [Sun Jul 08 16:21:13 CST 2018]; root of context hierarchy
bean銷毀...
由上例可以看出bean的生命周期為
- 通過構(gòu)造方法生成bean的實例
- 為bean注入屬性
- 調(diào)用初始化方法
- bean的使用
- IOC容器關(guān)閉時赌躺,調(diào)用銷毀方法
通過bean的后置處理器BeanPostProcessor
新建處理器類實現(xiàn)BeanPostProcessor接口
package com.cn.cmc.test;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
public class MyBeanPostProcessor implements BeanPostProcessor {
/**
* postProcessBeforeInitialization在init-method之后調(diào)用
* 參數(shù)arg0表示實例bean本身
* 參數(shù)arg1表示IOC容器中已配置bean的名字
* 返回值實際上為返回給用戶的bean
*/
@Override
public Object postProcessAfterInitialization(Object arg0, String arg1) throws BeansException {
// TODO Auto-generated method stub
System.out.println("postProcessAfterInitialization"+arg0+" "+arg1);
return arg0;
}
/**
* postProcessAfterInitialization在init-method之前調(diào)用
*/
@Override
public Object postProcessBeforeInitialization(Object arg0, String arg1) throws BeansException {
// TODO Auto-generated method stub
System.out.println("postProcessBeforeInitialization"+arg0+" "+arg1);
return arg0;
}
}
在配置文件中配置BeanPostProcessor
<bean class="com.cn.cmc.test.MyBeanPostProcessor"></bean>
1
由上例可以看出加了后置處理器之后執(zhí)行流程為:
- 通過構(gòu)造方法生成bean的實例
- 為bean注入屬性
- 將bean傳給后置處理器的postProcessBeforeInitialization方法
- 調(diào)用初始化方法
- 將bean傳給后置處理器的postProcessAfterInitialization方法
- bean的使用
- IOC容器關(guān)閉時,調(diào)用銷毀方法
————————————————
版權(quán)聲明:本文為CSDN博主「葉清逸」的原創(chuàng)文章羡儿,遵循 CC 4.0 BY-SA 版權(quán)協(xié)議礼患,轉(zhuǎn)載請附上原文出處鏈接及本聲明。
原文鏈接:https://blog.csdn.net/u013634252/article/details/80946404