一 Spring的概述:
1 概念:
- 學習Spring框架之前,我們學習了Hibernate框架(持久層的ORM框架)和Struts2(web層的MVC框架),spring框架叫做EE/SE開發的一站式框架,每一層都為我們提供了解決方案。
2 學習spring框架有啥好處?
- 面向對象的設計比任何技術都重要
- 方便解耦,簡化開發。
- 代碼測試更加簡單
- 方便集成各種優秀的框架
- 聲明式事務的支持
二 IOC
1 什么是IOC
IOC,Inversion of Control,控制反轉,是面向對象編程的一種設計思想。
原文地址:http://jinnianshilongnian.iteye.com/blog/1413846
1.2、IoC是什么
Ioc,在Java開發中,Ioc意味着將你設計好的對象交給容器控制,而不是傳統的在你的對象內部直接控制。如何理解好Ioc呢?理解好Ioc的關鍵是要明確“誰控制誰,控制什么,為何是反轉(有反轉就應該有正轉了),哪些方面反轉了”,那我們來深入分析一下:
●誰控制誰,控制什么:傳統Java SE程序設計,我們直接在對象內部通過new進行創建對象,是程序主動去創建依賴對象;而IoC是有專門一個容器來創建這些對象,即由Ioc容器來控制對 象的創建;誰控制誰?當然是IoC 容器控制了對象;控制什么?那就是主要控制了外部資源獲取(不只是對象包括比如文件等)。
●為何是反轉,哪些方面反轉了:有反轉就有正轉,傳統應用程序是由我們自己在對象中主動控制去直接獲取依賴對象,也就是正轉;而反轉則是由容器來幫忙創建及注入依賴對象;為何是反轉?因為由容器幫我們查找及注入依賴對象,對象只是被動的接受依賴對象,所以是反轉;哪些方面反轉了?依賴對象的獲取被反轉了。
圖1-2有IoC/DI容器后程序結構示意圖
1.2、IoC能做什么
IoC 不是一種技術,只是一種思想,一個重要的面向對象編程的法則,它能指導我們如何設計出松耦合、更優良的程序。傳統應用程序都是由我們在類內部主動創建依賴對象,從而導致類與類之間高耦合,難於測試;有了IoC容器后,把創建和查找依賴對象的控制權交給了容器,由容器進行注入組合對象,所以對象與對象之間是 松散耦合,這樣也方便測試,利於功能復用,更重要的是使得程序的整個體系結構變得非常靈活。
其實IoC對編程帶來的最大改變不是從代碼上,而是從思想上,發生了“主從換位”的變化。應用程序原本是老大,要獲取什么資源都是主動出擊,但是在IoC/DI思想中,應用程序就變成被動的了,被動的等待IoC容器來創建並注入它所需要的資源了。
IoC很好的體現了面向對象設計法則之一—— 好萊塢法則:“別找我們,我們找你”;即由IoC容器幫對象找相應的依賴對象並注入,而不是由對象主動去找。
1.3、IoC和DI
DI—Dependency Injection,即“依賴注入”:組件之間依賴關系由容器在運行期決定,形象的說,即由容器動態的將某個依賴關系注入到組件之中。依賴注入的目的並非為軟件系統帶來更多功能,而是為了提升組件重用的頻率,並為系統搭建一個靈活、可擴展的平台。通過依賴注入機制,我們只需要通過簡單的配置,而無需任何代碼就可指定目標需要的資源,完成自身的業務邏輯,而不需要關心具體的資源來自何處,由誰實現。
理解DI的關鍵是:“誰依賴誰,為什么需要依賴,誰注入誰,注入了什么”,那我們來深入分析一下:
●誰依賴於誰:當然是應用程序依賴於IoC容器;
●為什么需要依賴:應用程序需要IoC容器來提供對象需要的外部資源;
●誰注入誰:很明顯是IoC容器注入應用程序某個對象,應用程序依賴的對象;
●注入了什么:就是注入某個對象所需要的外部資源(包括對象、資源、常量數據)。
IoC和DI由什么關系呢?其實它們是同一個概念的不同角度描述,由於控制反轉概念比較含糊(可能只是理解為容器控制對象這一個層面,很難讓人想到誰來維護對象關系),所以2004年大師級人物Martin Fowler又給出了一個新的名字:“依賴注入”,相對IoC 而言,“依賴注入”明確描述了“被注入對象依賴IoC容器配置依賴對象”。
看過很多對Spring的Ioc理解的文章,好多人對Ioc和DI的解釋都晦澀難懂,反正就是一種說不清,道不明的感覺,讀完之后依然是一頭霧水,感覺就是開濤這位技術牛人寫得特別通俗易懂,他清楚地解釋了IoC(控制反轉) 和DI(依賴注入)中的每一個字,讀完之后給人一種豁然開朗的感覺。我相信對於初學Spring框架的人對Ioc的理解應該是有很大幫助的。
2 Spring Framework Runtime
3 實現的代碼如下:
面向接口編程的好處:方便程序的擴展(比如集合),多態
好的程序設計滿足OCP原則,在盡量不修改程序源代碼的基礎上進行擴展,
工廠模式:工廠+反射+配置文件實現解耦和
1 將實現類交給Spring管理
1 <?xml version="1.0" encoding="UTF-8"?> 2 <beans xmlns="http://www.springframework.org/schema/beans" 3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" 4 xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> 5 <bean id="UserService" class="com.itheima.demo1.UserServiceImpl"> 6 <property name="name" value="花花" /> 7 </bean> 8 </beans>
2 編寫測試類
/** * src目錄下的 */ @Test public void demo2(){ ApplicationContext applicationContext=new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userService = (UserService) applicationContext.getBean("UserService"); userService.save(); }
配置文件可以放在任意盤符下
/** * 配置文件放在D盤目錄下 */ @Test public void demo3(){ ApplicationContext applicationContext=new FileSystemXmlApplicationContext("D:\\applicationContext.xml"); UserService userService = (UserService) applicationContext.getBean("UserService"); userService.save(); }
4 IOC和DI(面試經常會問到)
IOC:控制反轉,將創建對象權交給spring管理
DI:依賴注入,前提必須有IOC的環境,Spring 管理的這個類的時候將類的屬性注入進來
面向對象的三種關系
(1)依賴
class A(){}
Class B(){public void xxx(A a){}}我們這時候說B依賴A
(2)繼承
(3)聚合
Spring工廠類
ApplicationCotext類的UML類圖
四 Bean的相關配置
1 <bean>標簽中的id和name
id :使用了約束中的唯一約束
name :沒有使用約束中的唯一約束(理論上可以出現重復,但是在實際的開發中不能重復)
2 Bean的作用范圍的配置(重點)
scope :Bean的作用范圍
singleton :默認的,Spring會采用單例模式創建這個對象。
prototype :多例模式。(Struts2和Spring整合一定會用到)
request :應用在web項目中,Spring創建這個類以后,將這個類存入到request范圍中。
session :應用在web項目中,Spring創建這個類以后,將這個類存入到session范圍中。
globalsession :應用在web項目中,必須在porlet環境下使用。但是如果沒有這種環境,相對於session。
3 屬性注入的方式
(1)使用構造方法注入
<!--spring屬性注入的方式 --> <!--使用構造方法--> <bean id="car" class="com.itheima.demo3.Car"> <constructor-arg name="name" value="瑪莎拉蒂"/> <constructor-arg name="price" value="2000000"/> </bean>
(2)使用set方法注入普通類型
<!--使用set方法注入普通類型--> <bean id="car2" class="com.itheima.demo3.Car2"> <property name="name" value="一汽大眾"/> <property name="price" value="200000"/> </bean>
使用set方法注入對象
<!--使用set方法注入對象--> <bean name="employee" class="com.itheima.demo3.Employee"> <property name="name" value="花花"/> <!--ref寫對象的id或者name--> <property name="car2" ref="car2"/> </bean>
(3)使用p名稱空間的方式(這點IDEA做的就比較好,不用去其他文件去找,直接快速修復)
<bean id="car2" class="com.itheima.demo3.Car2" p:name="奔馳" p:price="500000"/> <bean id="employee" class="com.itheima.demo3.Employee" p:name="大花花" p:car2-ref="car2"> </bean>
(4)使用SPEL表達式注入
<!--改用spel表達式注入--> <bean id="car2" class="com.itheima.demo3.Car2" > <property name="name" value="#{carInfo.name}"/> <property name="price" value="#{carInfo.jisuanprice()}"/> </bean> <bean id="employee" class="com.itheima.demo3.Employee"> <property name="name" value="#{'大花'}"/> <property name="car2" value="#{car2}" /><!--用spel不用ref,都用value--> </bean> <import resource="applicationContext2.xml"/>
擴展:集合類型屬性的注入
1 <?xml version="1.0" encoding="UTF-8"?> 2 <beans xmlns="http://www.springframework.org/schema/beans" 3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" 4 xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> 5 <!--注入數組--> 6 <bean id="Collection" class="com.itheima.demo4.Collection"> 7 <property name="arrs" > 8 <list> 9 <value>楊冪</value> 10 <value>楊紫</value> 11 <value>花花</value> 12 <value>李四</value> 13 </list> 14 </property> 15 <!--注入list集合--> 16 <property name="list" > 17 18 <list> 19 <value>張三</value> 20 <value>李四</value> 21 <value>王五</value> 22 </list> 23 </property> 24 <!--注入set集合--> 25 <property name="set"> 26 27 <set> 28 <value>aaa</value> 29 <value>bbb</value> 30 <value>xxx</value> 31 </set> 32 </property> 33 <!--注入Map集合--> 34 <property name="map"> 35 36 <map> 37 <entry key="aa" value="11"/> 38 <entry key="bb" value="22"/> 39 <entry key="cc" value="33"/> 40 </map> 41 </property> 42 </bean> 43 </beans>
養成良好的代碼注釋習慣,方便后期的查看。
往期內容: