主頁 > 後端開發 > spring的自動注入

spring的自動注入

2022-05-06 07:59:06 後端開發

Spring自動注入

spring的ioc

在剛開始學習spring的時候肯定都知道spring的兩個特點:ioc,aop,控制反轉和切面編程,這篇就只說說ioc

ioc是什么:在我們原來的代碼中,如果A依賴了B,那么我們會自己在A類中來new B,創建B的實體來使用,是程式主動的去創建依賴,但是我們在使用spring的了之后還會在A中主動的去創建B嗎?基本不會,因為創建物件的這個操作從原來的我們來控制變成了spring來管理,這個程序就稱為控制反轉,ioc并不是一種技術,而是一種編程思想,代碼的設計思路

那么這樣做的好處是什么?

  • 解耦,將物件之間的依賴關系交給spring來處理,避免硬編碼導致高度耦合

  • 資源的集中易于管理和配置

而spring實作ioc使用的方法是通過DI(依賴注入),當我們大部分的物件都被spring管理后那么spring也需要將我們A中所依賴的B,C,D...都給填充到A中,這個程序是由spring來管理的,我們只需要按照spring的規則宣告或指定,那么spring也會幫我們完成依賴的注入

自動注入

了解完spring的基本思想后,來回想一下當時學習spring的入門,有沒有說過spring的一個特點就是可以自動注入?

@Component
public class A   {
	@Autowired
	private B b;
}
@Component
public class B {
}

簡單的一段代碼,將A,B交給spring管理,在A中屬性b上添加一個@Autowired,當我們從spring的容器中取出A后發現屬性b居然有值,這個不就是自動注入嗎?

我是認為添加@Autowired注解是不算自動注入的,原因如下

1.名詞解釋

首先,什么叫自動(給翻譯翻譯什么叫tm的自動),生活中肯定都見過自動門,通過過自動門,當我們靠近的時候門會自動打開,通過后門自動關閉,自動就是指我們不需要手動的去開門/關門,那么這里的自動注入也是,我們不需要去手動的在需要注入的屬性上添加一個@Autowired注解

2.官方檔案

https://docs.spring.io/spring-framework/docs/current/reference/html/

在官方檔案的注入方式中,能看到

Dependency injection (DI) is a process whereby objects define their dependencies (that is, the other objects with which they work) only through constructor arguments, arguments to a factory method, or properties that are set on the object instance after it is constructed or returned from a factory method. The container then injects those dependencies when it creates the bean. This process is fundamentally the inverse (hence the name, Inversion of Control) of the bean itself controlling the instantiation or location of its dependencies on its own by using direct construction of classes or the Service Locator pattern.

Code is cleaner with the DI principle, and decoupling is more effective when objects are provided with their dependencies. The object does not look up its dependencies and does not know the location or class of the dependencies. As a result, your classes become easier to test, particularly when the dependencies are on interfaces or abstract base classes, which allow for stub or mock implementations to be used in unit tests.

DI exists in two major variants: Constructor-based dependency injection and Setter-based dependency injection.

翻譯就是

依賴項注入(DI)是一個程序,物件僅通過建構式引數、工廠方法的引數或物件實體構造或從工廠方法回傳后設定的屬性來定義其依賴項(即與它們一起作業的其他物件),然后,容器在創建bean時注入這些依賴項,這個程序基本上是bean本身的逆程序(因此稱為控制反轉),通過使用類的直接構造或服務定位器模式來控制其依賴項的實體化或位置,

使用DI原則,代碼更干凈,當物件具有依賴關系時,解耦更有效,物件不查找其依賴項,也不知道依賴項的位置或類別,因此,您的類變得更容易測驗,尤其是當依賴項位于介面或抽象基類上時,這允許在單元測驗中使用存根或模擬實作,

DI有兩種主要變體:基于建構式的依賴項注入和基于Setter的依賴項注入,

注意最后一段話:有兩種主要的變體:基于構造和setter方法的依賴注入,也就是說還有其他的注入方式,下面再細說

那么你可能會問?我需要怎么指定或宣告讓他去使用構造或setter方法進行注入而不是我手動指定@Autowired呢?

自動注入模式

還是spring的官網,網頁翻譯的

可以看到spring的自動裝配模式有下面的4種,而默認的是no,也就是不自動注入

現在來說一下setter注入/構造注入和自動注入模式的關系

假設我們需要在大學中找一個人,我們可以問老師/同學,或者去查看學生名單

那么找誰呢?我們可以去問他的名字,手機號,或者根據事件去問

比如我問老師有沒有一個叫張三的學生,可以得到結果,或者我根據手機號去查看學生的名單,也可以得到結果

我也可以去問同學,昨天下午逃課出去上網吧的人是誰?這就是根據事件去問,但是我能根據這個事件去查學生名單找到具體那個人嗎?不能

和spring的一樣,我可以根據型別去注入,通過的是setter方法,根據名稱去注入,也是通過的setter方法

但是當我指定使用構造注入的時候,那么就是通過構造方法進行注入

那么會發現自動注入以及DI的主要實作這里面并沒有所說的 "@Autowired"

那么我們怎么使用自動注入呢?

在xml配置中,只需要將頭定義中加上一句

default-autowire="byType"

<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
                           http://www.springframework.org/schema/beans/spring-beans.xsd"
       default-autowire="byType">

而在javaconfig中要麻煩一點,我們需要自定義一個配置類,實作BeanFactoryPostProcessorBeanDefinitionRegisterPostProcessor來修改beanDefinition的注入模型

@Component
public class BeanFactoryPostProcessor implements BeanFactoryPostProcessor {

	@Override
	public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
		ScannedGenericBeanDefinition a =(ScannedGenericBeanDefinition) beanFactory.getBeanDefinition("a");
		a.setAutowireMode(2);
	}
}

這個2是啥意思呢,在介面AutowireCapableBeanFactory中,spring定義了每個注入模型的值

public interface AutowireCapableBeanFactory extends BeanFactory {
	int AUTOWIRE_NO = 0;

	int AUTOWIRE_BY_NAME = 1;

	int AUTOWIRE_BY_TYPE = 2;

	int AUTOWIRE_CONSTRUCTOR = 3;
	@Deprecated
	int AUTOWIRE_AUTODETECT = 4;
    ....
}

那么當我們設定了byName或byType后,只需要提供一個setter方法即可,那么只要這個屬性名稱的bean存在于spring容器中就會被注入

public interface Test {
}
@Component
public class TestA implements Test {
}
@Component
public class A {

	@Autowired
	private Test testA;

	public void setTestA(Test testA) {
		System.out.println("走set方法啦");
		this.testA = testA;
	}

	public A() {
		System.out.println("走無參構造方法啦");
	}

	public A(Test testA) {
		System.out.println("走有參構造方法啦");
		this.testA = testA;
	}

	@Override
	public String toString() {
		return "A{" +
				"testA=" + testA +
				'}';
	}
}

結果:

走無參構造方法啦
走set方法啦
A{testA=com.jame.pojo.test.TestA@721e0f4f}

當我設定為byType后結果一樣,就不再粘貼代碼了

而當我設定為根據構造注入后,將注入的模型改成3

@Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        ScannedGenericBeanDefinition a =(ScannedGenericBeanDefinition) beanFactory.getBeanDefinition("a");
        a.setAutowireMode(3);
    }
}

結果

走有參構造方法啦
A{testA=com.jame.pojo.test.TestA@28864e92}

那么現在你會明白了所謂的@Autowired并不是自動注入,只要指定了這個bean的注入模型為byType/byName/構造后才是自動注入

@Autowired

@Autowired是使用setter注入是構造注入呢?是使用byName還是byType呢?

回答第一個問題:@Autowired是使用setter注入是構造注入呢?

將我的配置類BeanFactoryPostProcessor上的@Component注解去掉,然后在Test testA添加@Autowired

@Component
public class A {


	@Autowired
	private Test testA;

    //下面的代碼和上面粘貼出的的代碼一樣

}

結果

走無參構造方法啦
A{testA=com.jame.pojo.test.TestA@546a03af}

??什么情況,setter和構造都沒走,因為@Autowired底層使用的反射filed.set()來填充的屬性

DI有兩種主要變體:基于建構式的依賴項注入和基于Setter的依賴項注入

主要的是使用構造和setter,而其他的說的就是這種@Autowired的注入方式

第二個問題:是使用byName還是byType呢?

答案是兩個都是,或者兩個都不是,來看例子

public interface Test {
}
@Component
public class TestA implements Test {
}
@Component
public class A {
	@Autowired
	private Test test;//注意這里的屬性為test
    
    ....省略
}

這樣寫然后從spring容器中獲取肯定能獲取到結果,A中的testA屬性肯定有值的,就不演示了

但是,我給Test介面添加一個實作類TestB

@Component
public class TestB implements Test{

}

繼續執行代碼,發現報錯了

Error creating bean with name 'a': Unsatisfied dependency expressed through field 'test'; nested exception is org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'com.jame.pojo.test.Test' available: expected single matching bean but found 2: testA,testB
創建名稱為“a”的 bean 時出錯:通過欄位“test”表示的依賴關系不滿足;嵌套例外是 org.springframework.beans.factory.NoUniqueBeanDefinitionException:沒有“com.jame.pojo.test.Test”型別的合格 bean 可用:預期單個匹配 bean,但找到 2:testA,testB

到這里是不是就可以說明@Autowired是byType呢?,那怎么證明它也是byName呢?

修改A中的Test test屬性為testA時

@Component
public class A {


	@Autowired
	private Test testA;//注意這里的屬性為testA

	.....省略
}

繼續執行代碼發現又可以了

走無參構造方法啦
A{testA=com.jame.pojo.test.TestA@28864e92}

我們把屬性改為Test testB后繼續測驗

走無參構造方法啦
A{testB=com.jame.pojo.test.TestB@28864e92}

發現也是可以的,那么結論就是:@Autowired既是byType也是byName

當Spring容器中只有一個匹配的類時,會根據Type直接注入,而存在多個匹配的時候(接收的屬性定義為介面,有多個實作類),會直接拋出例外

這時候我們可以使用@Qualifier("testA")來指定具體哪一個類來注入

或者修改屬性的名字為需要注入類的名稱(首字母小寫)

測驗了一下@Qualifier()是高于byName的,也就是說即使存在兩個匹配的類,即使屬性名叫testB,我只要使用@Qualifier("testA")來指定bean,那么注入的就是testA

蕪湖沒了,拜拜

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

標籤:其他

上一篇:python是什么?作業前景如何?怎么算有基礎?爬資料違法嘛?。。

下一篇:分享一下 Idea 的 scope 功能

標籤雲
其他(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