Spring基礎知識彙總 Java開發必看
Spring簡介
Spring框架由Rod Johnson開發,2004年釋出了Spring框架的第一版。Spring是一個從實際開發中抽取出來的框架,因此它完成了大量開發中的通用步驟,留給開發者的僅僅是與特定應用相關的部分,從而大大提高了企業應用的開發效率。
Spring總結起來優點如下
- 低侵入式設計,程式碼的汙染極低
- 獨立於各種應用伺服器,基於Spring框架的應用,可以真正實現Write Once,Run Anywhere的承諾
- Spring的IoC容器降低了業務物件替換的複雜性,提高了元件之間的解耦
- Spring的AOP支援允許將一些通用任務如安全、事務、日誌等進行集中式管理,從而提供了更好的複用
- Spring的ORM和DAO提供了與第三方持久層框架的良好整合,並簡化了底層的資料庫訪問
- Spring的高度開放性,並不強制應用完全依賴於Spring,開發者可自由選用Spring框架的部分或全部
Spring框架的組成結構圖如下所示
spring-overview
Spring的核心機制
管理Bean
程式主要是通過Spring容器來訪問容器中的Bean,ApplicationContext是Spring容器最常用的介面,該介面有如下兩個實現類
- ClassPathXmlApplicationContext: 從類載入路徑下搜尋配置檔案,並根據配置檔案來建立Spring容器
- FileSystemXmlApplicationContext: 從檔案系統的相對路徑或絕對路徑下去搜尋配置檔案,並根據配置檔案來建立Spring容器
public class BeanTest{ public static void main(String args[]) throws Exception{ ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml"); Person p = ctx.getBean("person", Person.class); p.say(); } }
Eclipse使用Spring
在Eclipse等IDE工具中,使用者可以自建User Library
,然後把Spring的Jar包都放入其中,當然也可以將Jar包直接放在專案的/WEB-INF/lib
目錄下,但是如果使用User Library
,在專案釋出時,需要將使用者庫所引用的Jar檔案隨應用一起釋出,就是將User Library所使用的Jar複製到/WEB-INF/lib
目錄下,這是因為對於一個Web應用,Eclipse部署Web應用時不會將使用者庫的Jar檔案複製到/WEB-INF/lib
下,需要手動複製。
依賴注入
Spring框架的核心功能有兩個
- Spring容器作為超級大工廠,負責建立、管理所有的Java物件,這些Java物件被稱為Bean
- Spring容器管理容器中Bean之間的依賴關係,Spring使用一種被稱為“依賴注入”的方式來管理Bean之間的依賴關係
使用依賴注入,不僅可以為Bean注入普通的屬性值,還可以注入其他Bean的引用。依賴注入是一種優秀的解耦方式,其可以讓Bean以配置檔案組織在一起,而不是以硬編碼的方式耦合在一起。
理解依賴注入
Rod Johnson是第一個高度重視以配置檔案來管理Java例項的協作關係的人,他給這種方式起了一個名字:控制反轉(Inverse of Control,IoC)。後來Martine Fowler為這種方式起了另一個名稱:依賴注入(Dependency Injection),因此不管是依賴注入,還是控制反轉,其含義完全相同。當某個Java物件(呼叫者)需要呼叫另一個Java物件(被依賴物件)的方法時,在傳統模式下通常有兩種做法
- 原始做法: 呼叫者主動建立被依賴物件,然後再呼叫被依賴物件的方法
- 簡單工廠模式: 呼叫者先找到被依賴物件的工廠,然後主動通過工廠去獲取被依賴物件,最後再呼叫被依賴物件的方法
注意上面的主動二字,這必然會導致呼叫者與被依賴物件實現類的硬編碼耦合,非常不利於專案升級的維護。使用Spring框架之後,呼叫者無需主動獲取被依賴物件,呼叫者只要被動接受Spring容器為呼叫者的成員變數賦值即可,由此可見,使用Spring後,呼叫者獲取被依賴物件的方式由原來的主動獲取,變成了被動接受——所以Rod Johnson稱之為控制反轉。
另外從Spring容器的角度來看,Spring容器負責將被依賴物件賦值給呼叫者的成員變數——相當於為呼叫者注入它依賴的例項,因此Martine Fowler稱之為依賴注入。
設值注入
設值注入是指IoC容器通過成員變數的setter方法來注入被依賴物件。這種注入方式簡單、直觀,因而在Spring的依賴注入裡大量使用。
構造注入
利用構造器來設定依賴關係的方式,被稱為構造注入。通俗來說,就是驅動Spring在底層以反射方式執行帶指定引數的構造器,當執行帶引數的構造器時,就可利用構造器引數對成員變數執行初始化——這就是構造注入的本質。
兩種注入方式的對比
設值注入有如下優點
- 與傳統的JavaBean的寫法更相似,程式開發人員更容易理解、接受。通過setter方法設定依賴關係顯得更加直觀、自然
- 對於複雜的依賴關係,如果採用構造注入,會導致構造器過於臃腫,難以閱讀。Spring在建立Bean例項時,需要同時例項化其依賴的全部例項,因而導致效能下降。而使用設值注入,則能避免這些問題。
- 尤其在某些成員變數可選的情況下,多引數的構造器更加笨重
構造注入優勢如下
- 構造注入可以在構造器中決定依賴關係的注入順序,優先依賴的優先注入
- 對於依賴關係無需變化的Bean,構造注入更有用處。因為沒有setter方法,所有的依賴關係全部在構造器內設定,無須擔心後續的程式碼對依賴關係產生破壞
- 依賴關係只能在構造器中設定,則只有元件的建立者才能改變元件的依賴關係,對元件的呼叫者而言,元件內部的依賴關係完全透明,更符合高內聚的原則
Notes
建議採用設值注入為主,構造注入為輔的注入策略。對於依賴關係無須變化的注入,儘量採用構造注入;而其他依賴關係的注入,則考慮採用設值注入。
Spring容器中的Bean
對於開發者來說,開發者使用Spring框架主要是做兩件事:①開發Bean;②配置Bean。對於Spring框架來說,它要做的就是根據配置檔案來建立Bean例項,並呼叫Bean例項的方法完成“依賴注入”——這就是所謂IoC的本質。
容器中Bean的作用域
當通過Spring容器建立一個Bean例項時,不僅可以完成Bean例項的例項化,還可以為Bean指定特定的作用域。Spring支援如下五種作用域
- singleton: 單例模式,在整個Spring IoC容器中,singleton作用域的Bean將只生成一個例項
- prototype: 每次通過容器的getBean()方法獲取prototype作用域的Bean時,都將產生一個新的Bean例項
- request: 對於一次HTTP請求,request作用域的Bean將只生成一個例項,這意味著,在同一次HTTP請求內,程式每次請求該Bean,得到的總是同一個例項。只有在Web應用中使用Spring時,該作用域才真正有效
- 對於一次HTTP會話,session作用域的Bean將只生成一個例項,這意味著,在同一次HTTP會話內,程式每次請求該Bean,得到的總是同一個例項。只有在Web應用中使用Spring時,該作用域才真正有效
- global session: 每個全域性的HTTP Session對應一個Bean例項。在典型的情況下,僅在使用portlet context的時候有效,同樣只在Web應用中有效
如果不指定Bean的作用域,Spring預設使用singleton作用域。prototype作用域的Bean的建立、銷燬代價比較大。而singleton作用域的Bean例項一旦建立成果,就可以重複使用。因此,應該儘量避免將Bean設定成prototype作用域。
使用自動裝配注入合作者Bean
Spring能自動裝配Bean與Bean之間的依賴關係,即無須使用ref顯式指定依賴Bean,而是由Spring容器檢查XML配置檔案內容,根據某種規則,為呼叫者Bean注入被依賴的Bean。
Spring自動裝配可通過<beans/>
元素的default-autowire
屬性指定,該屬性對配置檔案中所有的Bean起作用;也可通過對<bean/>
元素的autowire
屬性指定,該屬性只對該Bean起作用。
autowire
和default-autowire
可以接受如下值
no
: 不使用自動裝配。Bean依賴必須通過ref元素定義。這是預設配置,在較大的部署環境中不鼓勵改變這個配置,顯式配置合作者能夠得到更清晰的依賴關係byName
: 根據setter方法名進行自動裝配。Spring容器查詢容器中全部Bean,找出其id與setter方法名去掉set字首,並小寫首字母后同名的Bean來完成注入。如果沒有找到匹配的Bean例項,則Spring不會進行任何注入byType
: 根據setter方法的形參型別來自動裝配。Spring容器查詢容器中的全部Bean,如果正好有一個Bean型別與setter方法的形參型別匹配,就自動注入這個Bean;如果找到多個這樣的Bean,就丟擲一個異常;如果沒有找到這樣的Bean,則什麼都不會發生,setter方法不會被呼叫constructor
: 與byType類似,區別是用於自動匹配構造器的引數。如果容器不能恰好找到一個與構造器引數型別匹配的Bean,則會丟擲一個異常autodetect
: Spring容器根據Bean內部結構,自行決定使用constructor或byType策略。如果找到一個預設的建構函式,那麼就會應用byType策略
當一個Bean既使用自動裝配依賴,又使用ref顯式指定依賴時,則顯式指定的依賴覆蓋自動裝配依賴;對於大型的應用,不鼓勵使用自動裝配。雖然使用自動裝配可減少配置檔案的工作量,但大大將死了依賴關係的清晰性和透明性。依賴關係的裝配依賴於原始檔的屬性名和屬性型別,導致Bean與Bean之間的耦合降低到程式碼層次,不利於高層次解耦
<!--通過設定可以將Bean排除在自動裝配之外--> <bean id="" autowire-candidate="false"/> <!--除此之外,還可以在beans元素中指定,支援模式字串,如下所有以abc結尾的Bean都被排除在自動裝配之外--> <beans default-autowire-candidates="*abc"/>
建立Bean的3種方式
使用構造器建立Bean例項
使用構造器來建立Bean例項是最常見的情況,如果不採用構造注入,Spring底層會呼叫Bean類的無引數構造器來建立例項,因此要求該Bean類提供無引數的構造器。
採用預設的構造器建立Bean例項,Spring對Bean例項的所有屬性執行預設初始化,即所有的基本型別的值初始化為0或false;所有的引用型別的值初始化為null。
使用靜態工廠方法建立Bean
使用靜態工廠方法建立Bean例項時,class屬性也必須指定,但此時class屬性並不是指定Bean例項的實現類,而是靜態工廠類,Spring通過該屬性知道由哪個工廠類來建立Bean例項。
除此之外,還需要使用factory-method屬性來指定靜態工廠方法,Spring將呼叫靜態工廠方法返回一個Bean例項,一旦獲得了指定Bean例項,Spring後面的處理步驟與採用普通方法建立Bean例項完全一樣。如果靜態工廠方法需要引數,則使用<constructor-arg.../>
元素指定靜態工廠方法的引數。
呼叫例項工廠方法建立Bean
例項工廠方法與靜態工廠方法只有一個不同:呼叫靜態工廠方法只需使用工廠類即可,而呼叫例項工廠方法則需要工廠例項。使用例項工廠方法時,配置Bean例項的<bean.../>
元素無須class屬性,配置例項工廠方法使用factory-bean
指定工廠例項。
採用例項工廠方法建立Bean的<bean.../>
元素時需要指定如下兩個屬性
- factory-bean: 該屬性的值為工廠Bean的id
- factory-method: 該屬性指定例項工廠的工廠方法
若呼叫例項工廠方法時需要傳入引數,則使用<constructor-arg.../>
元素確定引數值。
協調作用域不同步的Bean
當singleton作用域的Bean依賴於prototype作用域的Bean時,會產生不同步的現象,原因是因為當Spring容器初始化時,容器會預初始化容器中所有的singleton Bean
,由於singleton Bean
依賴於prototype Bean
,因此Spring在初始化singleton Bean
之前,會先建立prototypeBean
——然後才建立singleton Bean
,接下里將prototype Bean
注入singleton Bean
。
解決不同步的方法有兩種
- 放棄依賴注入: singleton作用域的Bean每次需要prototype作用域的Bean時,主動向容器請求新的Bean例項,即可保證每次注入的
prototype Bean
例項都是最新的例項 - 利用方法注入: 方法注入通常使用lookup方法注入,使用lookup方法注入可以讓Spring容器重寫容器中Bean的抽象或具體方法,返回查詢容器中其他Bean的結果,被查詢的Bean通常是一個
non-singleton Bean
。Spring通過使用JDK動態代理或cglib庫修改客戶端的二進位制碼,從而實現上述要求
建議採用第二種方法,使用方法注入。為了使用lookup方法注入,大致需要如下兩步
- 將呼叫者Bean的實現類定義為抽象類,並定義一個抽象方法來獲取被依賴的Bean
- 在
<bean.../>
元素中新增<lookup-method.../>
子元素讓Spring為呼叫者Bean的實現類實現指定的抽象方法
Notes
Spring會採用執行時動態增強的方式來實現
<lookup-method.../>
元素所指定的抽象方法,如果目標抽象類實現過介面,Spring會採用JDK動態代理來實現該抽象類,併為之實現抽象方法;如果目標抽象類沒有實現過介面,Spring會採用cglib實現該抽象類,併為之實現抽象方法。Spring4.0的spring-core-xxx.jar包中已經整合了cglib類庫。
兩種後處理器
Spring提供了兩種常用的後處理器
- Bean後處理器: 這種後處理器會對容器中Bean進行後處理,對Bean進行額外加強
- 容器後處理器: 這種後處理器會對IoC容器進行後處理,用於增強容器功能
Bean後處理器
Bean後處理器是一種特殊的Bean,這種特殊的Bean並不對外提供服務,它甚至可以無須id屬性,它主要負責對容器中的其他Bean執行後處理,例如為容器中的目標Bean生成代理等,這種Bean稱為Bean後處理器。Bean後處理器會在Bean例項建立成功之後,對Bean例項進行進一步的增強處理。Bean後處理器必須實現BeanPostProcessor
介面,同時必須實現該介面的兩個方法。
Object postProcessBeforeInitialization(Object bean, String name) throws BeansException
: 該方法的第一個引數是系統即將進行後處理的Bean例項,第二個引數是該Bean的配置idObject postProcessAfterinitialization(Object bean, String name) throws BeansException
: 該方法的第一個引數是系統即將進行後處理的Bean例項,第二個引數是該Bean的配置id
容器中一旦註冊了Bean後處理器,Bean後處理器就會自動啟動,在容器中每個Bean建立時自動工作,Bean後處理器兩個方法的回撥時機如下圖
bean-post-process
注意一點,如果使用BeanFactory
作為Spring容器,則必須手動註冊Bean後處理器,程式必須獲取Bean後處理器例項,然後手動註冊。
BeanPostProcessor bp = (BeanPostProcessor)beanFactory.getBean("bp"); beanFactory.addBeanPostProcessor(bp); Person p = (Person)beanFactory.getBean("person");
容器後處理器
Bean後處理器負責處理容器中的所有Bean例項,而容器後處理器則負責處理容器本身。容器後處理器必須實現BeanFactoryPostProcessor
介面,並實現該介面的一個方法postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory)
實現該方法的方法體就是對Spring容器進行的處理,這種處理可以對Spring容器進行自定義擴充套件,當然也可以對Spring容器不進行任何處理。
類似於BeanPostProcessor
,ApplicationContext
可自動檢測到容器中的容器後處理器,並且自動註冊容器後處理器。但若使用BeanFactory
作為Spring容器,則必須手動呼叫該容器後處理器來處理BeanFactory
容器。
Spring的“零配置”支援
搜尋Bean類
Spring提供如下幾個Annotation來標註Spring Bean
@Component
: 標註一個普通的Spring Bean類@Controller
: 標註一個控制器元件類@Service
: 標註一個業務邏輯元件類@Repository
: 標註一個DAO元件類
在Spring配置檔案中做如下配置,指定自動掃描的包
<context:component-scan base-package="edu.shu.spring.domain"/>
使用@Resource配置依賴
@Resource
位於javax.annotation
包下,是來自JavaEE規範的一個Annotation
,Spring直接借鑑了該Annotation
,通過使用該Annotation
為目標Bean指定協作者Bean。使用@Resource
與<property.../>
元素的ref屬性有相同的效果。
@Resource
不僅可以修飾setter方法,也可以直接修飾例項變數,如果使用@Resource
修飾例項變數將會更加簡單,此時Spring將會直接使用JavaEE規範的Field注入,此時連setter方法都可以不要。
使用@PostConstruct和@PreDestroy定製生命週期行為
@PostConstruct
和@PreDestroy
同樣位於javax.annotation包下,也是來自JavaEE規範的兩個Annotation,Spring直接借鑑了它們,用於定製Spring容器中Bean的生命週期行為。它們都用於修飾方法,無須任何屬性。其中前者修飾的方法時Bean的初始化方法;而後者修飾的方法時Bean銷燬之前的方法。
Spring4.0增強的自動裝配和精確裝配
Spring提供了@Autowired
註解來指定自動裝配,@Autowired
可以修飾setter方法、普通方法、例項變數和構造器等。當使用@Autowired
標註setter方法時,預設採用byType自動裝配策略。在這種策略下,符合自動裝配型別的候選Bean例項常常有多個,這個時候就可能引起異常,為了實現精確的自動裝配,Spring提供了@Qualifier
註解,通過使用@Qualifier
,允許根據Bean的id來執行自動裝配。
Spring的AOP
為什麼需要AOP
AOP(Aspect Orient Programming)也就是面向切面程式設計,作為物件導向程式設計的一種補充,已經成為一種比較成熟的程式設計方式。其實AOP問世的時間並不太長,AOP和OOP互為補充,面向切面程式設計將程式執行過程分解成各個切面。
AOP專門用於處理系統中分佈於各個模組(不同方法)中的交叉關注點的問題,在JavaEE應用中,常常通過AOP來處理一些具有橫切性質的系統級服務,如事務管理、安全檢查、快取、物件池管理等,AOP已經成為一種非常常用的解決方案。
使用AspectJ實現AOP
AspectJ是一個基於Java語言的AOP框架,提供了強大的AOP功能,其他很多AOP框架都借鑑或採納其中的一些思想。其主要包括兩個部分:一個部分定義瞭如何表達、定義AOP程式設計中的語法規範,通過這套語法規範,可以方便地用AOP來解決Java語言中存在的交叉關注點的問題;另一個部分是工具部分,包括編譯、除錯工具等。
AOP實現可分為兩類
- 靜態AOP實現: AOP框架在編譯階段對程式進行修改,即實現對目標類的增強,生成靜態的AOP代理類,以AspectJ為代表
- 動態AOP實現: AOP框架在執行階段動態生成AOP代理,以實現對目標物件的增強,以Spring AOP為代表
一般來說,靜態AOP實現具有較好的效能,但需要使用特殊的編譯器。動態AOP實現是純Java實現,因此無須特殊的編譯器,但是通常效能略差。
AOP的基本概念
關於面向切面程式設計的一些術語
- 切面(Aspect): 切面用於組織多個Advice,Advice放在切面中定義
- 連線點(Joinpoint): 程式執行過程中明確的點,如方法的呼叫,或者異常的丟擲。在Spring AOP中,連線點總是方法的呼叫
- 增強處理(Advice): AOP框架在特定的切入點執行的增強處理。處理有“around”、“before”和“after”等型別
- 切入點(Pointcut): 可以插入增強處理的連線點。簡而言之,當某個連線點滿足指定要求時,該連線點將被新增增強處理,該連線點也就變成了切入點
Spring的AOP支援
Spring中的AOP代理由Spring的IoC容器負責生成、管理,其依賴關係也由IoC容器負責管理。
為了在應用中使用@AspectJ
支援,Spring需要新增三個庫
aspectjweaver.jar
aspectjrt.jar
aopalliance.jar
並在Spring配置檔案中做如下配置
<!--啟動@AspectJ支援--> <aop:aspectj-autoproxy/> <!--指定自動搜尋Bean元件、自動搜尋切面類--> <context:component-scan base-package="edu.shu.sprint.service"> <context:include-filter type="annotation" expression="org.aspectj.lang.annotation.Aspect"/> </context:component-scan>
相關文章
- linux 基礎知識彙總Linux
- Java常見知識點彙總(③)——物件導向基礎Java物件
- 作為一名Java開發者應該掌握的基礎知識彙總!Java
- Java基礎知識總結Java
- JS基礎知識梳理彙總 如何能學好Web前端開發JSWeb前端
- Java多執行緒程式設計基礎知識彙總Java執行緒程式設計
- Linux磁碟管理基礎知識全彙總Linux
- Java基礎知識總結-1Java
- Java基礎知識點總結Java
- 【docker基礎知識】docker坑問題彙總Docker
- 小白必看 | 代理ip伺服器知識彙總伺服器
- Java併發--基礎知識Java
- Java基礎對反射知識總結Java反射
- 藍橋杯知識點彙總:基礎知識和常用演算法演算法
- Spring知識總結1:spring開發流程Spring
- java基礎知識Java
- Java基礎面試知識點總結Java面試
- 【知識分享】 伺服器基礎知識【初學者必看】伺服器
- 初識Java Java基礎知識Java
- Spring Cloud:Consul基礎知識SpringCloud
- 前端開發 JavaScript 乾貨知識點彙總前端JavaScript
- Java個人知識點總結(基礎篇)Java
- 多執行緒面試必備基礎知識彙總執行緒面試
- Java常見知識點彙總(⑳)——鎖Java
- Java常見知識點彙總(②)——StaticJava
- Java知識彙總——思維導圖Java
- Java基礎概念知識Java
- java基礎知識點Java
- Java基礎知識(二)Java
- Html與css基礎知識介紹(必看篇)HTMLCSS
- 【Java面試題系列】:Java基礎知識常見面試題彙總 第二篇Java面試題
- Rust 基礎知識總結Rust
- SpringIOC基礎知識總結Spring
- 索引基礎知識總結索引
- React 基礎知識總結React
- 知識點-Spring Boot 異常處理彙總Spring Boot
- 前端知識彙總前端
- Docker 知識彙總Docker
- MySQL知識彙總MySql