javascript
Spring Boot教程(7) – 直观地理解Spring容器
在你學習Spring之前,你肯定聽說過“控制反轉”、“依賴注入”、“上下文”等名詞,伴隨著這些名詞的,是一些冗長晦澀的解釋,這些解釋并沒有什么顯著的錯誤,但是因為太過抽象導致初學者們無法直觀地去理解它們。
廢話不多說,我們通過舉例或者寫代碼來一步一步理解它們。
在一個系統(tǒng)里,可能有多個業(yè)務部分,比如用戶服務,訂單服務,數據服務等等,在代碼里,這些服務也許分布在多個類中,可能叫做UserService或者OrderService等等。為了簡便,我們假設系統(tǒng)里有A、B、C、D四個類,他們有如下的依賴關系:
簡單來說,D是B的成員,A和B是C的成員。以前Spring這種東西沒有出現(xiàn)的時候,你可能手動地去創(chuàng)建ABCD對象,再設置他們的依賴關系,如下圖:
上面的代碼看起來很簡單,也不復雜。問題是當你的系統(tǒng)規(guī)模擴大了以后,有上百個對象需要初始化以及設置依賴的時候,復雜度就直線上升了。
系統(tǒng)類的對象越多,依賴關系越復雜為了降低復雜度,減少他們之間的耦合,這個時候我們就需要用上Spring容器了。程序里的對象都可以扔到容器里,對象只需要告訴容器它需要哪些依賴就行,容器自動初始化好并給它。比如上面ABCD那個例子,C告訴容器它需要A和B,B告訴容器它需要D,這樣ABCD四個對象都可以扔到容器里了,當你想用C的時候,就從容器里面拿出來,這時候C的成員就已經包含了A和B了(我們目前只討論單例的情況,就是一個類只有一個對象)。
容器通過“配置”來了解對象之間的依賴關系。
在Spring框架里,ApplicationContext就代表了容器(又叫應用程序上下文),容器里的對象,又叫Bean。配置有兩種方式,一種是xml,一種是Java配置(或者說代碼配置)。早些年的時候,Java Web(SSH)開發(fā)為人詬病的,就是臃腫的xml配置,雖然目前Spring仍然支持xml配置和混合配置,不過Spring Boot已經建議使用Java配置了。我們來通過一個例子來看看如何使用容器和配置:
圖中顯示了項目的源碼結構、程序的入口、配置以及依賴。
上圖寫出了ABCD的源碼。
配置文件是Config.java,它有一個@Configuration注解,這表明Config類是配置類。它還有另外一個注解@ComponentScan,這個注解表示容器應該去掃描程序的代碼,看看那些組件應當被初始化為Bean。哪些類會被初始化為Bean呢?我們看看ABCD的源碼,發(fā)現(xiàn)他們各自都有注解@Component,這個注解就表示了它所在的類是組件,需要初始化Bean。C和B的成員變量都用@Autowired注解來修飾,這樣容器就知道他們都依賴哪些Bean了,ABCD的對象都生成好之后,容器就把D對象賦值給B的成員,把A和B的對象賦值給C的成員。
如果你運行程序,會發(fā)現(xiàn)輸出的結果是true,也就是C里的A成員不為空。這樣你是不是就理解了Spring容器的作用了呢?你的程序不必再在ABCD的外部去設置他們的依賴了,反而只需要在ABCD的類的內部指明依賴就行,這個從外而內的過程就是控制反轉(IoC,Inversion of Control),控制權交給了Spring容器,Spring容器也叫IoC容器。你還可能聽過依賴注入(DI,Dependency Injection),如果你看網上文章沒明白,權當他們是一樣的好了。
對于開源庫,他們的類肯定沒有@Component注解,這樣肯定不能通過掃描來加入容器。那就需要寫點代碼了。Config類中有一個objectMapper方法,返回了一個ObjectMapper對象(ObjectMapper是Jackson的關鍵類),objectMapper方法有個@Bean注解,表示返回的對象會加入到容器中。ObjectMapper對象可以在objectMapper方法里,進行一些修改呀,配置呀等等。
這個時候ObjectMapper對象還沒人用,如果ABCD任何一個類想用,只需要在類中加一個成員@Autowired ObjectMapper objectmapper;,Spring容器會自動把ObjectMapper對象賦值到這個成員上。非常方便,尤其是在大項目里,類非常多,一行代碼就把你需要的對象拿過來用,簡直爽的不行。在實踐中,依賴的類常常是個接口,使用者只調用接口就行,至于容器給它哪種實現(xiàn)并不重要,這個邏輯在“配置”里就可以控制。這樣,程序的耦合度進一步減小了。
在我們之前運行第一個Spring Boot應用的時候,我們曾經添加了一個控制器,他有個@Controller注解,這個注解相當于一個特殊的@Component,只不過@Controller表示當前的組件是用來處理網絡請求的。類似的還有,@Service用來表示業(yè)務相關的組件,@Repository用來表示數據獲取相關的組件。他們都是通過自動掃描放到Spring容器中的。
總結
為了讓大家理解Spring容器,本文忽略了很多技術上的細節(jié),比如Bean的命名,比如說@ComponentScan的掃描路徑和注解參數,比如聲明依賴除了在成員上加@Autowired以外還能通過構造方法或者setter等等。我相信你懂了Spring容器的概念之后,這些都不是什么問題。
博客鏈接:fookwood.com/spring-boot…
轉載于:https://juejin.im/post/5d106ab26fb9a07f03574621
總結
以上是生活随笔為你收集整理的Spring Boot教程(7) – 直观地理解Spring容器的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: swiper踩过的哪些坑
- 下一篇: WPF入门教程系列四——Dispatch