主頁 > 後端開發 > 一分鐘玩轉 Spring IoC!

一分鐘玩轉 Spring IoC!

2020-09-25 02:53:54 後端開發

前言

上一篇文章」我們對 Spring 有了初步的認識,而 Spring 全家桶中幾乎所有組件都是依賴于 IoC 的,

剛開始聽到 IoC,會覺得特別高大上,但其實掰開了很簡單,

跟著我的腳步,一文帶你吃透 IoC 原理,

本文主要講原理,圍繞“是何”、“為何”來談,下一篇文章會講實踐部分,也就是“如何”,

是何

上一篇文章有同學問我在官網該看哪些內容,怎么找的,那今天的截圖里都會有鏈接,

初識 IoC

根據上一篇文章我們說的,Spring 全家桶中最重要的幾個專案都是基于 Spring Framework 的,所以我們就以 Spring Framework 為例來看檔案,

首先它的右側有 Github 的鏈接,另外點到「LEARN」這里,就會看到各個版本的檔案,

那我們點「Reference Doc」,就能夠看到它的一些模塊的介紹:

(等下... 模塊?什么是模塊?這個問題下文回答,)

第一章 Overview,講述它的歷史、設計原理等等;

第二章 Core,包含了 IoC 容器,AOP 等等,那自然是講 Spring 的核心了,要點進去好好看了,

點進去之后發現了寶貴的學習資料,一切的 what, why, how 都可以在這里找到答案,

這里很好的解釋了大名鼎鼎的 IoC - Inversion of Control, 控制反轉,

每次讀都會有新的體會和識訓,

我粗略的總結一下:控制反轉就是把創建和管理 bean 的程序轉移給了第三方,而這個第三方,就是 Spring IoC Container,對于 IoC 來說,最重要的就是容器

容器負責創建、配置和管理 bean,也就是它管理著 bean 的生命,控制著 bean 的依賴注入,

通俗點講,因為專案中每次創建物件是很麻煩的,所以我們使用 Spring IoC 容器來管理這些物件,需要的時候你就直接用,不用管它是怎么來的、什么時候要銷毀,只管用就好了,

舉個例子,就好像父母沒時間管孩子,就把小朋友交給托管所,就安心的去上班而不用管孩子了,
托兒所,就是第三方容器,負責管理小朋友的吃喝玩樂;
父母,相當于程式員,只管接送孩子,不用管他們吃喝,

等下,bean 又是什么?

Bean 其實就是包裝了的 Object,無論是控制反轉還是依賴注入,它們的主語都是 object,而 bean 就是由第三方包裝好了的 object,(想一下別人送禮物給你的時候都是要包裝一下的,自己造的就免了,

IoC 容器

既然說容器是 IoC 最重要的部分,那么 Spring 如何設計容器的呢?
還是回到官網,第二段有介紹哦:

答:使用 ApplicationContext,它是 BeanFactory 的子類,更好的補充并實作了 BeanFactory 的,

BeanFactory 簡單粗暴,可以理解為 HashMap:

  • Key - bean name
  • Value - bean object

但它一般只有 get, put 兩個功能,所以稱之為“低級容器”,

ApplicationContext 多了很多功能,因為它繼承了多個介面,可稱之為“高級容器”,在下文的搭建專案中,我們會使用它,

ApplicationContext 的里面有兩個具體的實作子類,用來讀取配置配件的:

  • ClassPathXmlApplicationContext - 從 class path 中加載組態檔,更常用一些;
  • FileSystemXmlApplicationContext - 從本地檔案中加載組態檔,不是很常用,如果再到 Linux 環境中,還要改路徑,不是很方便,

當我們點開 ClassPathXmlApplicationContext 時,發現它并不是直接繼承 ApplicationContext 的,它有很多層的依賴關系,每層的子類都是對父類的補充實作,

而再往上找,發現最上層的 class 回到了 BeanFactory,所以它非常重要,

要注意,Spring 中還有個 FactoryBean,兩者并沒有特別的關系,只是名字比較接近,所以不要弄混了順序,

為了好理解 IoC,我們先來回顧一下不用 IoC 時寫代碼的程序,

深入理解 IoC

這里用經典 class Rectangle 來舉例:

  • 兩個變數:長和寬
  • 自動生成 set() 方法和 toString() 方法

注意 ??:一定要生成 set() 方法,因為 Spring IoC 就是通過這個 set() 方法注入的;
toString() 方法是為了我們方便列印查看,

public class Rectangle {
    private int width;
    private int length;

    public Rectangle() {
        System.out.println("Hello World!");
    }


    public void setWidth(int widTth) {
        this.width = widTth;
    }

    public void setLength(int length) {
        this.length = length;
    }

    @Override
    public String toString() {
        return "Rectangle{" +
                "width=" + width +
                ", length=" + length +
                '}';
    }
}

然后在 test 檔案中手動用 set() 方法給變數賦值,

嗯,其實這個就是「解藕」的程序!

public class MyTest {
  @Test
  public void myTest() {
    Rectangle rect = new Rectangle();
    rect.setLength(2);
    rect.setWidth(3);
    System.out.println(rect);
  }
}

其實這就是 IoC 給屬性賦值的實作方法,我們把「創建物件的程序」轉移給了 set() 方法,而不是靠自己去 new,就不是自己創建的了,

這里我所說的“自己創建”,指的是直接在物件內部來 new,是程式主動創建物件的正向的程序;
這里使用 set() 方法,是別人(test)給我的;
而 IoC 是用它的容器來創建、管理這些物件的,其實也是用的這個 set() 方法,不信,你把這個這個方法去掉或者改個名字試試?

幾個關鍵問題:

何為控制,控制的是什么?

答:是 bean 的創建、管理的權利,控制 bean 的整個生命周期,

何為反轉,反轉了什么?

答:把這個權利交給了 Spring 容器,而不是自己去控制,就是反轉,
由之前的自己主動創建物件,變成現在被動接收別人給我們的物件的程序,這就是反轉,

舉個生活中的例子,主動投資和被動投資,

自己炒股、選股票的人就是主動投資,主動權掌握在自己的手中;
而買基金的人就是被動投資,把主動權交給了基金經理,除非你把這個基金賣了,否則具體選哪些投資產品都是基金經理決定的,

依賴注入

回到檔案中,第二句話它說:IoC is also known as DI.

我們來談談 dependency injection - 依賴注入,

何為依賴,依賴什么?

程式運行需要依賴外部的資源,提供程式內物件的所需要的資料、資源,

何為注入,注入什么?

組態檔把資源從外部注入到內部,容器加載了外部的檔案、物件、資料,然后把這些資源注入給程式內的物件,維護了程式內外物件之間的依賴關系,

所以說,控制反轉是通過依賴注入實作的,
但是你品,你細品,它們是有差別的,像是「從不同角度描述的同一件事」

  • IoC 是設計思想,DI 是具體的實作方式;
  • IoC 是理論,DI 是實踐;

從而實作物件之間的解藕,

當然,IoC 也可以通過其他的方式來實作,而 DI 只是 Spring 的選擇,

IoC 和 DI 也并非 Spring 框架提出來的,Spring 只是應用了這個設計思想和理念到自己的框架里去,

為何

那么為什么要用 IoC 這種思想呢?換句話說,IoC 能給我們帶來什么好處?

答:解藕,

它把物件之間的依賴關系轉成用組態檔來管理,由 Spring IoC Container 來管理,

在專案中,底層的實作都是由很多個物件組成的,物件之間彼此合作實作專案的業務邏輯,但是,很多很多物件緊密結合在一起,一旦有一方出問題了,必然會對其他物件有所影響,所以才有了解藕的這種設計思想,

如上圖所示,本來 ABCD 是互相關聯在一起的,當加入第三方容器的管理之后,每個物件都和第三方法的 IoC 容器關聯,彼此之間不再直接聯系在一起了,沒有了耦合關系,全部物件都交由容器來控制,降低了這些物件的親密度,就叫“解藕”,

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/122648.html

標籤:Java

上一篇:小林求職記(二):說好的問基礎,為啥我感覺一點也不基礎呢?

下一篇:Spire.Cloud.Word 加密 Word 檔案

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more