Spring框架 第2章 IOC容器和Bean的配置

第2章 ?IOC容器和Bean的配置

2.1 IOC和DI

2.1.1 IOC(Inversion of Control):反轉(zhuǎn)控制

在應(yīng)用程序中的組件需要獲取資源時,傳統(tǒng)的方式是組件主動的從容器中獲取所需要的資源,在這樣的模式下開發(fā)人員往往需要知道在具體容器中特定資源的獲取方式,增加了學(xué)習(xí)成本,同時降低了開發(fā)效率。

反轉(zhuǎn)控制的思想完全顛覆了應(yīng)用程序組件獲取資源的傳統(tǒng)方式:反轉(zhuǎn)了資源的獲取方向——改由容器主動的將資源推送給需要的組件,開發(fā)人員不需要知道容器是如何創(chuàng)建資源對象的,只需要提供接收資源的方式即可,極大的降低了學(xué)習(xí)成本,提高了開發(fā)的效率。這種行為也稱為查找的被動形式。

2.1.2 DI(Dependency Injection):依賴注入

IOC的另一種表述方式:即組件以一些預(yù)先定義好的方式(例如:setter 方法)接受來自于容器的資源注入。相對于IOC而言,這種表述更直接。

2.1.3 IOC容器在Spring中的實現(xiàn)

1)在通過IOC容器讀取Bean的實例之前,需要先將IOC容器本身實例化。

2)Spring提供了IOC容器的兩種實現(xiàn)方式

① BeanFactory:IOC容器的基本實現(xiàn),是Spring內(nèi)部的基礎(chǔ)設(shè)施,是面向

Spring本身的,不是提供給開發(fā)人員使用的。

② ApplicationContext:BeanFactory的子接口,提供了更多高級特性。面向Spring的使用者,幾乎所有場合都使用ApplicationContext而不是底層的BeanFactory。

2.1.4 ApplicationContext的主要實現(xiàn)類

? 1)? ClassPathXmlApplicationContext:對應(yīng)類路徑下的XML格式的配置文件

? 2)? FileSystemXmlApplicationContext:對應(yīng)文件系統(tǒng)中的XML格式的配置文件

? 3)? 在初始化時就創(chuàng)建單例的bean,也可以通過配置的方式指定創(chuàng)建的Bean是多實例的。

2.1.5 ConfigurableApplicationContext

? 1)? 是ApplicationContext的子接口,包含一些擴展方法

? 2)? refresh()和close()讓ApplicationContext具有啟動、關(guān)閉和刷新上下文的能力。

2.1.6 WebApplicationContext

? 1)? 專門為WEB應(yīng)用而準(zhǔn)備的,它允許從相對于WEB根目錄的路徑中完成初始化工作

2.2 通過類型獲取bean

? 1)? 從IOC容器中獲取bean時,除了通過id值獲取,還可以通過bean的類型獲取。但如果同一個類型的bean在XML文件中配置了多個,則獲取時會拋出異常,所以同一個類型的bean在容器中必須是唯一的。? 2)? 或者可以使用另外一個重載的方法,同時指定bean的id值和類型

2.3 給bean的屬性賦值

2.3.1 依賴注入的方式

1. 通過bean的setXxx()方法賦值

Hello World中使用的就是這種方式

2. 通過bean的構(gòu)造器賦值

? 1)? Spring自動匹配合適的構(gòu)造器? 2)? 通過索引值指定參數(shù)位置? 3)? 通過類型區(qū)分重載的構(gòu)造器

2.3.2 ?p名稱空間

為了簡化XML文件的配置,越來越多的XML文件采用屬性而非子元素配置信息。Spring 從2.5版本開始引入了一個新的p命名空間,可以通過<bean>元素屬性的方式配置Bean 的屬性。

使用p命名空間后,基于XML的配置方式將進一步簡化。

2.3.3 可以使用的值

1. 字面量

? 1)? 可以使用字符串表示的值,可以通過value屬性或value子節(jié)點的方式指定

? 2)? 基本數(shù)據(jù)類型及其封裝類、String等類型都可以采取字面值注入的方式

? 3)? 若字面值中包含特殊字符,可以使用<![CDATA[]]>把字面值包裹起來

2. null值

3. 給bean的級聯(lián)屬性賦值

4. 外部已聲明的bean

5. 內(nèi)部bean

當(dāng)bean實例僅僅給一個特定的屬性使用時,可以將其聲明為內(nèi)部bean。內(nèi)部bean聲明直接包含在<property>或<constructor-arg>元素里,不需要設(shè)置任何id或name屬性

內(nèi)部bean不能使用在任何其他地方

2.4 集合屬性

在Spring中可以通過一組內(nèi)置的XML標(biāo)簽來配置集合屬性,例如:<list>,<set>或<map>。

2.4.1 數(shù)組和List

配置java.util.List類型的屬性,需要指定<list>標(biāo)簽,在標(biāo)簽里包含一些元素。這些標(biāo)簽 可以通過<value>指定簡單的常量值,通過<ref>指定對其他Bean的引用。通過<bean> 指定內(nèi)置bean定義。通過<null/>指定空元素。甚至可以內(nèi)嵌其他集合。

數(shù)組的定義和List一樣,都使用<list>元素。

配置java.util.Set需要使用<set>標(biāo)簽,定義的方法與List一樣。

2.4.2 Map

Java.util.Map通過<map>標(biāo)簽定義,<map>標(biāo)簽里可以使用多個<entry>作為子標(biāo)簽。每個條目包含一個鍵和一個值。

必須在<key>標(biāo)簽里定義鍵。

因為鍵和值的類型沒有限制,所以可以自由地為它們指定<value>、<ref>、<bean>或<null/>元素。

可以將Map的鍵和值作為<entry>的屬性定義:簡單常量使用key和value來定義;bean引用通過key-ref和value-ref屬性定義。

2.4.3 集合類型的bean

如果只能將集合對象配置在某個bean內(nèi)部,則這個集合的配置將不能重用。我們需要 將集合bean的配置拿到外面,供其他bean引用。

配置集合類型的bean需要引入util名稱空間

2.5 ?FactoryBean

2.5.1 FactoryBean

Spring中有兩種類型的bean,一種是普通bean,另一種是工廠bean,即FactoryBean。

工廠bean跟普通bean不同,其返回的對象不是指定類的一個實例,其返回的是該工 廠bean的getObject方法所返回的對象。

工廠bean必須實現(xiàn)org.springframework.beans.factory.FactoryBean接口。

2.6 bean的高級配置

2.6.1 配置信息的繼承

1. 背景

查看下面兩個Employee的配置,其中dept屬性是重復(fù)的。

2. 配置信息的繼承

Spring允許繼承bean的配置,被繼承的bean稱為父bean。繼承這個父bean的bean 稱為子bean

子bean從父bean中繼承配置,包括bean的屬性配置

子bean也可以覆蓋從父bean繼承過來的配置

3. 補充說明

父bean可以作為配置模板,也可以作為bean實例。若只想把父bean作為模板,可以設(shè)置<bean>的abstract 屬性為true,這樣Spring將不會實例化這個bean

如果一個bean的class屬性沒有指定,則必須是抽象bean

并不是<bean>元素里的所有屬性都會被繼承。比如:autowire,abstract等。

也可以忽略父bean的class屬性,讓子bean指定自己的類,而共享相同的屬性配置。 但 此時abstract必須設(shè)為true。

2.6.2 bean之間的依賴

有的時候創(chuàng)建一個bean的時候需要保證另外一個bean也被創(chuàng)建,這時我們稱前面的bean對后面的bean有依賴。例如:要求創(chuàng)建Employee對象的時候必須創(chuàng)建Department。 這里需要注意的是依賴關(guān)系不等于引用關(guān)系,Employee即使依賴Department也可以不引用它。

2.7 ?bean的作用域

在Spring中,可以在<bean>元素的scope屬性里設(shè)置bean的作用域,以決定這個bean是單實例的還是多實例的。

默認(rèn)情況下,Spring只為每個在IOC容器里聲明的bean創(chuàng)建唯一一個實例,整個IOC容器范圍內(nèi)都能共享該實例:所有后續(xù)的getBean()調(diào)用和bean引用都將返回這個唯一的bean實例。該作用域被稱為singleton,它是所有bean的默認(rèn)作用域。

當(dāng)bean的作用域為單例時,Spring會在IOC容器對象創(chuàng)建時就創(chuàng)建bean的對象實例。而當(dāng)bean的作用域為prototype時,IOC容器在獲取bean的實例時創(chuàng)建bean的實例對象。

2.8 ?bean的生命周期

? 1)? Spring IOC容器可以管理bean的生命周期,Spring允許在bean生命周期內(nèi)特定的時間點執(zhí)行指定的任務(wù)。

? 2)? Spring IOC容器對bean的生命周期進行管理的過程:

① 通過構(gòu)造器或工廠方法創(chuàng)建bean實例

② 為bean的屬性設(shè)置值和對其他bean的引用

③ 調(diào)用bean的初始化方法

④ bean可以使用了

⑤?當(dāng)容器關(guān)閉時,調(diào)用bean的銷毀方法

? 3)? 在配置bean時,通過init-method和destroy-method 屬性為bean指定初始化和銷毀方法

? 4)? bean的后置處理器

① bean后置處理器允許在調(diào)用初始化方法前后對bean進行額外的處理

② bean后置處理器對IOC容器里的所有bean實例逐一處理,而非單一實例。其典型 ???應(yīng)用是:檢查bean屬性的正確性或根據(jù)特定的標(biāo)準(zhǔn)更改bean的屬性。

③ bean后置處理器時需要實現(xiàn)接口:

org.springframework.beans.factory.config.BeanPostProcessor。在初始化方法被調(diào)用前

后,Spring將把每個bean實例分別傳遞給上述接口的以下兩個方法:

  • postProcessBeforeInitialization(Object, String)
  • postProcessAfterInitialization(Object, String)

? 5)? 添加bean后置處理器后bean的生命周期

①通過構(gòu)造器或工廠方法創(chuàng)建bean實例

②為bean的屬性設(shè)置值和對其他bean的引用

③將bean實例傳遞給bean后置處理器的postProcessBeforeInitialization()方法

④調(diào)用bean的初始化方法

⑤將bean實例傳遞給bean后置處理器的postProcessAfterInitialization()方法

⑥bean可以使用了

⑦當(dāng)容器關(guān)閉時調(diào)用bean的銷毀方法

2.9 引用外部屬性文件

當(dāng)bean的配置信息逐漸增多時,查找和修改一些bean的配置信息就變得愈加困難。這時可以將一部分信息提取到bean配置文件的外部,以properties格式的屬性文件保存起來,同時在bean的配置文件中引用properties屬性文件中的內(nèi)容,從而實現(xiàn)一部分屬性值在發(fā)生變化時僅修改properties屬性文件即可。這種技術(shù)多用于連接數(shù)據(jù)庫的基本信息的配置。

2.9.1 直接配置

2.9.2 使用外部的屬性文件

1. 創(chuàng)建properties屬性文件

2. 引入context名稱空間

3.指定properties屬性文件的位置

4.從properties屬性文件中引入屬性值

2.10 自動裝配

2.10.1 自動裝配的概念

? 1? )? 自動裝配:根據(jù)指定的裝配規(guī)則,不需要明確指定,Spring自動將匹配的屬性值注入bean中。

2.10.2 裝配模式

? 1)? 根據(jù)類型自動裝配:將類型匹配的bean作為屬性注入到另一個bean中。若IOC容器中有多個與目標(biāo)bean類型一致的bean,Spring將無法判定哪個bean最合適該屬性,所以不能執(zhí)行自動裝配

? 2)? 根據(jù)名稱自動裝配:必須將目標(biāo)bean的名稱和屬性名設(shè)置的完全相同

? 3)? 通過構(gòu)造器自動裝配:當(dāng)bean中存在多個構(gòu)造器時,此種自動裝配方式將會很復(fù)雜。不推薦使用。

2.10.3 選用建議

相對于使用注解的方式實現(xiàn)的自動裝配,在XML文檔中進行的自動裝配略顯笨拙,在項目中更多的使用注解的方式實現(xiàn)。

2.11 通過注解配置bean

2.11.1 概述

相對于XML方式而言,通過注解的方式配置bean更加簡潔和優(yōu)雅,而且和MVC組件化開發(fā)的理念十分契合,是開發(fā)中常用的使用方式。

2.11.2 使用注解標(biāo)識組件

? 1)? 普通組件:@Component

標(biāo)識一個受Spring IOC容器管理的組件

? 2)? 持久化層組件:@Repository

標(biāo)識一個受Spring IOC容器管理的持久化層組件

? 3)? 業(yè)務(wù)邏輯層組件:@Service

標(biāo)識一個受Spring IOC容器管理的業(yè)務(wù)邏輯層組件

? 4)? 表述層控制器組件:@Controller

標(biāo)識一個受Spring IOC容器管理的表述層控制器組件

? 5)? 組件命名規(guī)則

①默認(rèn)情況:使用組件的簡單類名首字母小寫后得到的字符串作為bean的id

②使用組件注解的value屬性指定bean的id

注意:事實上Spring并沒有能力識別一個組件到底是不是它所標(biāo)記的類型,即使將 @Respository注解用在一個表述層控制器組件上面也不會產(chǎn)生任何錯誤,所以 @Respository、@Service、@Controller這幾個注解僅僅是為了讓開發(fā)人員自己明確 當(dāng)前的組件扮演的角色。

2.11.3 ?掃描組件

組件被上述注解標(biāo)識后還需要通過Spring進行掃描才能夠偵測到。

? 1)? 指定被掃描的package

<context:component-scan?base-package="com.atguigu.component"/>

? 2)? 詳細(xì)說明

base-package屬性指定一個需要掃描的基類包,Spring容器將會掃描這個基類包及其

子包中的所有類。

②當(dāng)需要掃描多個包時可以使用逗號分隔。

③如果僅希望掃描特定的類而非基包下的所有類,可使用resource-pattern屬性過濾特定的類,示例:

<context:component-scan?

base-package="com.atguigu.component"?

resource-pattern="autowire/*.class"/>

④包含與排除

  • <context:include-filter>子節(jié)點表示要包含的目標(biāo)類

注意:通常需要與use-default-filters屬性配合使用才能夠達到“僅包含某些 組件”這樣的效果。即:通過將use-default-filters屬性設(shè)置為false, 禁用默認(rèn)過濾器,然后掃描的就只是include-filter中的規(guī)則指定的 組件了。

  • <context:exclude-filter>子節(jié)點表示要排除在外的目標(biāo)類
  • component-scan下可以擁有若干個include-filter和exclude-filter子節(jié)點
  • 過濾表達式

類別

示例

說明

annotation

com.atguigu.XxxAnnotation

過濾所有標(biāo)注了XxxAnnotation的類。這個規(guī)則根據(jù)目標(biāo)組件是否標(biāo)注了指定類型的注解進行過濾。

assignable

com.atguigu.BaseXxx

過濾所有BaseXxx類的子類。這個規(guī)則根據(jù)目標(biāo)組件是否是指定類型的子類的方式進行過濾。

aspectj

com.atguigu.*Service+

所有類名是以Service結(jié)束的,或這樣的類的子類。這個規(guī)則根據(jù)AspectJ表達式進行過濾。

regex

com\.atguigu\.anno\.*

所有com.atguigu.anno包下的類。這個規(guī)則根據(jù)正則表達式匹配到的類名進行過濾。

custom

com.atguigu.XxxTypeFilter

使用XxxTypeFilter類通過編碼的方式自定義過濾規(guī)則。該類必須實現(xiàn)org.springframework.core.type.filter.TypeFilter接口

? 3)? JAR包

必須在原有JAR包組合的基礎(chǔ)上再導(dǎo)入一個:spring-aop-4.0.0.RELEASE.jar

2.11.4 組件裝配

? 1)? 需求

Controller組件中往往需要用到Service組件的實例,Service組件中往往需要用到 Repository組件的實例。Spring可以通過注解的方式幫我們實現(xiàn)屬性的裝配。

? 2)? 實現(xiàn)依據(jù)

在指定要掃描的包時,<context:component-scan> 元素會自動注冊一個bean的后置處 理器:AutowiredAnnotationBeanPostProcessor的實例。該后置處理器可以自動裝配標(biāo)記 了@Autowired、@Resource或@Inject注解的屬性。

? 3)? @Autowired注解

①根據(jù)類型實現(xiàn)自動裝配。

②構(gòu)造器、普通字段(即使是非public)、一切具有參數(shù)的方法都可以應(yīng)用@Autowired ??注解

③默認(rèn)情況下,所有使用@Autowired注解的屬性都需要被設(shè)置。當(dāng)Spring找不到匹 ???配的bean裝配屬性時,會拋出異常。

④若某一屬性允許不被設(shè)置,可以設(shè)置@Autowired注解的required屬性為 false

⑤默認(rèn)情況下,當(dāng)IOC容器里存在多個類型兼容的bean時,Spring會嘗試匹配bean ??的id值是否與變量名相同,如果相同則進行裝配。如果bean的id值不相同,通過類????????????? ??型的自動裝配將無法工作。此時可以在@Qualifier注解里提供bean的名稱。Spring ??甚至允許在方法的形參上標(biāo)注@Qualifiter注解以指定注入bean的名稱。 ⑥@Autowired注解也可以應(yīng)用在數(shù)組類型的屬性上,此時Spring將會把所有匹配的bean進行自動裝配。

⑦@Autowired注解也可以應(yīng)用在集合屬性上,此時Spring讀取該集合的類型信息,然后自動裝配所有與之兼容的bean。

⑧@Autowired注解用在java.util.Map上時,若該Map的鍵值為String,那么 Spring將自動裝配與值類型兼容的bean作為值,并以bean的id值作為鍵。

? 4)? @Resource

@Resource注解要求提供一個bean名稱的屬性,若該屬性為空,則自動采用標(biāo)注處的變量或方法名作為bean的名稱。

? 5)? @Inject

@Inject和@Autowired注解一樣也是按類型注入匹配的bean,但沒有reqired屬性。