看完這篇你還敢說,不懂Spring中的IoC容器?

木子莫 發佈 2020-02-10T02:46:15+00:00

一.什麼是IoC1. 什麼是耦合和內聚耦合指的就是模塊之間的依賴關係。public class SpringConfiguration {

一. 什麼是IoC

1. 什麼是耦合和內聚

  • 耦合指的就是模塊之間的依賴關係。模塊間的依賴越多,則表示耦合度越高,相應的維護成本就越高。
  • 內聚指的是模塊內功能之間的聯繫。模塊內功能的聯繫越緊密,則表示內聚度越高,模塊的職責也就越單一。

所以在程序開發中應該儘量的降低耦合,提高內聚。也就是設計原則中的開閉原則和單一職責原則。

2. 工廠模式

工廠模式就是用來解決程序間耦合的一種設計模式。可以把所有要創建的對象放在工廠的一個集合里,當需要使用這個對象的時候,直接從工廠裡面取出來用就行。

工廠模式的優點:

  • 一個調用者想創建一個對象,只需要指定相應的名字即可從工廠中獲得這個對象。
  • 屏蔽了產品的具體實現,調用者只關心產品的接口。

3. 控制反轉(IoC)

控制反轉在維基百科中的定義:

控制反轉(Inversion of Control,縮寫為IoC),是面向對象編程中的一種設計原則,可以用來減低計算機代碼之間的耦合度。其中最常見的方式叫做依賴注入(Dependency Injection,簡稱DI),還有一種方式叫「依賴查找」(Dependency Lookup)。通過控制反轉,對象在被創建的時候,由一個調控系統內所有對象的外界實體,將其所依賴的對象的引用傳遞(注入)給它。

下面再從控制和反轉兩個詞分兩個方面來理解:

  • 誰控制誰?IoC容器控制了對象。控制什麼?控制了對象要獲取的外部資源(其它對象或數據等)
  • 什麼是反轉?是IoC容器查找並注入依賴給對象,對象是被動的接受,而不是主動的創建,所以是反轉。

通過new方式來主動獲取對象:

通過IoC容器獲取對象(注意看箭頭的方向,是不是反轉了):

有了IoC容器後,把創建和查找依賴對象的控制權交給了容器,由容器進行注入組合對象,所以對象與對象之間是鬆散耦合,這樣也方便測試,利於功能復用,更重要的是使得程序的整個體系結構變得非常靈活。

4. 依賴注入(DI)

依賴注入在維基百科中的定義

在軟體工程中,依賴注入是種實現控制反轉用於解決依賴性設計模式。一個依賴關係指的是可被利用的一種對象(即服務提供端) 。依賴注入是將所依賴的傳遞給將使用的從屬對象(即客戶端)。該服務是將會變成客戶端的狀態的一部分。傳遞服務給客戶端,而非允許客戶端來建立或尋找服務,是本設計模式的基本要求。

其實依賴注入和控制反轉表達的是一個意思。控制反轉是一種思想,而依賴注入是這個思想的最典型的實現方法。

由IoC來控制對象的依賴,通過構造函數、變量或Setter等方法來將依賴注入到對象中,這樣就將對象和對象的依賴進行了解耦。

二. spring中的工廠類

我們常用的spring容器是ApplicationContext,先來看一下它的依賴結構。

由圖可知,spring容器中的頂層接口是BeanFactory。ApplicationContext是它的子接口(注意這個也是個接口哦)。它默認一讀取配置文件,就會創建對象放到容器中。再來看一下ApplicationContext的三個主要的實現類。

  • ClassPathXmlApplication:它是從類的根路徑下加載xml配置文件(推薦用這種)。
  • FileSystemXmlApplication: 它是從磁碟路徑上加載配置文件,配置文件可以在磁碟的任意位置。(但使用不靈活,不推薦)
  • AnnotationConfigApplication:當我們使用註解配置容器對象時,需要使用此類來創建spring容器。它用來讀取註解。(springboot默認使用這個)

三. Bean的創建和管理

1. bean標籤

作用 : 用於配置對象讓spring來創建的。默認情況下它調用的是類中的無參構造函數。如果沒有無參構造函數則不能創建成功。

屬性:

  • id:給對象在容器中提供一一個唯一 標識。用於獲取對象。
  • class:指定類的全限定類名。用於反射創建對象。默認情況下調用無參構造函數。
  • scope:指定對象的作用範圍。
    • singleton : 單例對象,也是默認的。
    • prototype : 多例對象,每次都創建一個不同的對象。
    • request :WEB 項目中,Spring創建一個Bean的對象,將對象存入到request域中.
    • session : WEB項目中,Spring創建一個Bean的對象,將對象存入到session域中。
    • global session:WEB項目中,應用在集群環境.如果沒有集群環境那麼globalSession相當於session.
  • init-method:指定類中的初始化方法名稱。
  • destroy-method:指定類中銷毀方法名稱。

2. bean的作用範圍和生命周期

①. 單例對象: scope="singleton"一個應用只有一一個對象的實例。它的作用範圍就是整個引用。生命周期:

  • 對象出生:當應用加載,創建容器時,對象就被創建了。
  • 對象活著:只要容器在,對象-直活著。
  • 對象死亡:當應用卸載,銷毀容器時,對象就被銷毀了。

②. 多例對象: scope="prototype"每次訪問對象時,都會重新創建對象實例。生命周期:

  • 每次訪問對象時,都會重新創建對象實例。
  • 對象活著:只要對象在使用中,就一直活著。
  • 對象死亡:由java的垃圾回收器機制來處理。

四. spring中的註解

1. 用於創建對象的


相當於<bean id = "" class = "" />

①. @component

  • 作用:把資源讓spring來管理,相當於在xml中註冊一個bean。
  • 屬性:value:指定bean的id.如果不指定value屬性,默認bean的id是當前類的類名。首字母小寫。

②.@Service 、@Repository 、@Controller他們都是對@Component註解的衍生,其實作用是一模一樣的,只是提供了更明確的語義化。

  • @Repository:一般用於持久層的註解。
  • @Service:一般用於業務層的註解
  • @Controller:一般用於表現層的註解。

2. 用於注入數據的


相當於<property name = "" ref = "" />  或
      <property name= "" value = "" />

①. @Autowried

作用:自動按照類型注入。當使用註解注入屬性時,set 方法可以省略。它只能注入其他bean類型(ByType)。當有多個類型匹配時,使用要注入的對象變量名稱作為bean的id (ByName) ,在spring容器查找,找到了也可以注入成功。找不到就報錯。

三種注入方式(屬性輸入、Setter注入、構造函數注入):


/**
 * 使用變量注入依賴
 */
@Autowired
private IAccountDao accountDao ;

/**
 * 使用構造器注入 spring推薦使用這個
 * @param accountDao 要注入的依賴
 */
@Autowired
public AccountServiceImpl(IAccountDao accountDao) {
    this.accountDao = accountDao;
}

/**
 * 使用Setter注入
 * @param accountDao 要注入的依賴
 */
@Autowired
public void setAccountDao(IAccountDao accountDao) {
    this.accountDao = accountDao;
}

②. @Qualifier

  • 作用:在自動按照類型注入的基礎之上,再按照Bean的id注入。它在給欄位注入時不能獨立使用,必須和@Autowire一起使用;但是給方法參數注入時,可以獨立使用。
  • 屬性:value:指定bean的id.

③. @Resource

  • 作用:@Resource採用 name 屬性。默認情況下,Spring 將 value 解釋為要注入的 bean name。也就是ByName注入。
  • 屬性:value:指定bean的id.

④. @Value

  • 作用:用於注入基本類型數據和String類型數據
  • 屬性:用於指定值,可使用SpEL表達式。

3. 用於改變作用域的


相當於<bean id = "" class = "" scope = ""/>中的scope屬性

①. @Scope

  • 作用:指定bean的作用範圍。
  • 屬性:value:指定bean的作用範圍。取值: singleton prototype request session globalsession。

4. 生命周期相關的


相當於<bean id = "" class = "" init-method = "" destory-method = ""/>中的init-method屬性和destory-method屬性

①. @PostConstruct

作用:用於指定初始化方法

②. @PostDestory

作用:用於指定銷毀方法

5. spring中的新註解

①. @Configuration

  • 作用:用於指定當前類是一個spring配置類,當創建容器時會從該類上加載註解。獲取容器時需要使用AnnotationApplicationContext (有@Configuration註解的類. class)。
  • 屬性:value:用於指定配置類的字節碼

②. @ComponentScan

  • 作用:用於指定spring在初始化容器時要掃描的包。作用和在spring的xml配置文件中的<context : component-scan base-package="com. itheima"/>是一樣的。
  • 屬性:basePackages: 用於指定要掃描的包。和該註解中的value屬性作用一樣。

③. Bean

  • 作用:該註解只能寫在方法(該方法的返回值作為bean放到容器中)上,表明使用此方法創建一個對象, 並且放入spring容器。
  • 屬性:name:給當前@Bean註解方法創建的對象指定一個名稱 (即bean的id)。

④. @PropertySource

  • 作用:用於加載.properties文件中的配置。例如我們配置數據源時,可以把連接資料庫的信息寫到properties配置文件中,就可以使用此註解指定properties配置文件的位置。
  • 屬性:value[] :用於指定properties文件位置。如果是在類路徑下,需要寫上classpath:。

示例:


@Configuration
@PropertySource("classpath:jdbc.properties")
public class JdbcConfig{
}

⑤. @Import

  • 作用:用於導入其他配置類,在引入其他配置類時,可以不用再寫@Configuration註解。當然,寫上也沒問題。
  • 屬性:value[] :用干指定其他配置類的字節碼。

示例:


@Configuration
@ComponentScan(basePackages = "com.ncusoft.springDemos")
@Import({ JdbcConfig.class })
public class SpringConfiguration {
}

6. 選擇XML還是註解?

先來看一下各自的優勢:

  • 註解的優勢:配置簡單,維護方便(我們找到類,就相當於找到了對應的配置)。
  • XML的優勢:修改時,不用改源碼。不涉及重新編譯和部署。

應該根據實際的開發來選擇使用(springboot推薦使用註解),一般在source code(原始碼)中的類使用註解來創建bean(更方便,只需一個註解搞定)。從外部引入的依賴可選擇使用XML來創建bean。


關鍵字: