欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

spring依賴注入成功但在調(diào)用接口的時(shí)候拿到的依賴卻是null問題

 更新時(shí)間:2023年12月14日 09:42:44   作者:不喜歡吃貓的魚  
這篇文章主要介紹了spring依賴注入成功但在調(diào)用接口的時(shí)候拿到的依賴卻是null問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

前言

使用過spring的同學(xué)們都知道,如果出現(xiàn)了依賴注入失敗的問題首先會(huì)想到以下幾點(diǎn):

1、對(duì)應(yīng)的類有沒有寫@Service或@Component注解以供能被spring掃描注冊(cè);

2、在配置中有沒有配置要掃描的包路徑,或是對(duì)應(yīng)的類是否在已配置的包路徑下;

3、配置的包路徑錯(cuò)誤導(dǎo)致掃描失??;

4、同一個(gè)項(xiàng)目中jar沖突導(dǎo)致在注冊(cè)bean過程失敗進(jìn)而導(dǎo)致注入不成功;

問題描述

使用springboot的時(shí)候發(fā)現(xiàn)項(xiàng)目正常編譯運(yùn)行,所有的bean都被掃描加載了,但是在前端調(diào)用接口的時(shí)候發(fā)現(xiàn)controller里的service竟然是null。

具體如下圖所示:

具體截圖

由上圖可以看出Device對(duì)應(yīng)的那個(gè)bean在spring容器中已經(jīng)是被注冊(cè)過了的,就意味著并不是對(duì)應(yīng)的類沒有被加載掃描到。

但我通過在測(cè)試類中運(yùn)行發(fā)現(xiàn)使用同樣的注入方式是可以注入成功并成功運(yùn)行,如下圖所示:

注入測(cè)試

為了進(jìn)一步測(cè)試是否能正常的注入依賴,將原先的注入方式改成了構(gòu)造注入的方式,發(fā)現(xiàn)也是能夠正常的注入進(jìn)去,如下圖所示:

通過以上測(cè)試發(fā)現(xiàn),這次的問題并不是因?yàn)閽呙杪窂交蚴亲⒔獾膯栴}導(dǎo)致的,當(dāng)然這里的jar依賴這些也是正常的。

分析

仔細(xì)回想下這次的問題,項(xiàng)目服務(wù)能正常啟動(dòng),在spring的bean容器中也能找到對(duì)應(yīng)的bean那就說明bean注冊(cè)過程是沒有問題的。

接下來通過注入測(cè)試發(fā)現(xiàn)這幾個(gè)service也是能夠正常的注入到controller里,那說明項(xiàng)目中DI這個(gè)過程也是OK的,否則的話不可能是換一種注入方式就能注入。

由此可見,此問題并不是因?yàn)閟pring的問題導(dǎo)致。接下來本菜鳥又查詢了一些資料說可能是因?yàn)閯?dòng)態(tài)代理的問題導(dǎo)致,項(xiàng)目中沒有獨(dú)特申明代理方式所以默認(rèn)使用的是JDK的動(dòng)態(tài)代理。

轉(zhuǎn)眼一想感覺是發(fā)現(xiàn)什么了(此處省略一萬個(gè)滑稽…),感覺離成功又更近一步了。

這里我們先來看下常用的動(dòng)態(tài)代理:

1、jdk動(dòng)態(tài)代理:

  • 使用jdk動(dòng)態(tài)代理只能對(duì)實(shí)現(xiàn)了接口的類生成代理,而不能針對(duì)類;
  • 使用反射生成代理類;

2、cglib動(dòng)態(tài)代理:

  • 使用cglib代理是針對(duì)類實(shí)現(xiàn)代理,主要是對(duì)指定的類生成一個(gè)子類,覆蓋其中的方法(繼承);
  • 底層采用ASM字節(jié)碼生成框架,使用字節(jié)碼技術(shù)生成代理類,比使用Java反射效率要高。

了解到這里后本菜感覺想到點(diǎn)什么了,本菜的項(xiàng)目中沒有單獨(dú)設(shè)置代理模式,所以spring對(duì)bean處理的時(shí)候默認(rèn)是使用了jdk動(dòng)態(tài)代理,而項(xiàng)目里的service并非是接口而是一個(gè)class,而且還未實(shí)現(xiàn)某個(gè)接口,只是單純的class

?

本菜就想,是不是因?yàn)榻涌谑菃渭兊念悰]有實(shí)現(xiàn)接口,而jdk的動(dòng)態(tài)代理沒有將這個(gè)類代理到,所以導(dǎo)致注入失敗了,于是本菜就設(shè)置了下代理為cglib動(dòng)態(tài)代理:

但經(jīng)本菜鳥測(cè)試發(fā)現(xiàn),問題依舊存在…(內(nèi)心開始MMP了)。這時(shí)本菜鳥又想起了另一個(gè)點(diǎn),spring在對(duì)bean進(jìn)行代理的時(shí)候會(huì)自行選擇代理方式:

(1)當(dāng)Bean實(shí)現(xiàn)接口時(shí),Spring就會(huì)用JDK的動(dòng)態(tài)代理;

(2)當(dāng)Bean沒有實(shí)現(xiàn)接口時(shí),Spring就使用CGlib動(dòng)態(tài)代理;

所以說再次申明代理方式也是沒有用的。

除了以上的一些疑點(diǎn),本菜又發(fā)現(xiàn)出現(xiàn)問題的代碼里使用了spring的AOP,

于是開始懷疑是不是AOP搞鬼了,于是找了相關(guān)代碼

從這里可見這里AOP切面切點(diǎn)用在了調(diào)用controller方法前,進(jìn)而再聯(lián)想到項(xiàng)目目前的代理方式是cglib動(dòng)態(tài)代理而非jdk動(dòng)態(tài)代理。于是猜想是不是因?yàn)锳OP切入的時(shí)候代理未生效沒有將對(duì)應(yīng)的bean給到對(duì)應(yīng)的方法里,后面繼續(xù)深入查詢看看@Aspect使用的代理方式

鎖定這個(gè)

繼續(xù)深入,看到這里發(fā)現(xiàn)看到有ProxyCreator (代理創(chuàng)建)

判斷生成的bean是否能夠被代理

創(chuàng)建代理類bean實(shí)例

代碼較多省略了部分,這里繼續(xù)走到獲取代理

這里我們使用的是cglib動(dòng)態(tài)代理

關(guān)鍵代碼判斷對(duì)應(yīng)bean是否需要被代理并返回實(shí)例

設(shè)置過濾

中間的代碼比較多,省略了部分,關(guān)鍵代碼

這里可以看到對(duì)映射的方法進(jìn)行了過濾,依次看過去發(fā)現(xiàn)這里存在疑慮

看到這里就有疑問了,因?yàn)槭褂肁OP的時(shí)候?qū)υL問修飾符是沒有啥影響的,public、private、protect都是可以的,所以最終問題應(yīng)該是這里在使用cglib的時(shí)候把私有方法過濾掉了不被代理,所以導(dǎo)致了在controller中依賴注入是成功的但調(diào)用這個(gè)接口方法時(shí)拿到的依賴為null,回看項(xiàng)目源代碼:

解決問題

把這里的private改成public就可以了…

反思

論寫代碼仔細(xì)的重要性?。?!雖然此次的事故代碼不是出自我手,但是這個(gè)小小的問題卻耗費(fèi)了我好久時(shí)間,這里讓我不得不反思下以后寫代碼要仔細(xì)仔細(xì)再仔細(xì),同時(shí)遇到bug的時(shí)候在排查問題的時(shí)候也是要仔細(xì)仔細(xì)仔細(xì)!不然很簡(jiǎn)單的問題也是會(huì)被忽略掉。

總結(jié)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • SpringBoot項(xiàng)目集成xxljob實(shí)現(xiàn)全紀(jì)錄

    SpringBoot項(xiàng)目集成xxljob實(shí)現(xiàn)全紀(jì)錄

    XXL-JOB是一個(gè)分布式任務(wù)調(diào)度平臺(tái),本文主要介紹了SpringBoot項(xiàng)目集成xxljob實(shí)現(xiàn)全紀(jì)錄,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • Java ThreadLocal的詳細(xì)解釋

    Java ThreadLocal的詳細(xì)解釋

    這篇文章主要介紹了快速了解Java中ThreadLocal類,介紹了ThreadLocal 是什么,ThreadLocal的作用,ThreadLocal 原理等相關(guān)內(nèi)容,具有一定參考價(jià)值,需要的朋友可以了解下。
    2021-11-11
  • IDEA集成DeepSeek通過離線安裝解決無法安裝Proxy?AI插件問題(最新推薦)

    IDEA集成DeepSeek通過離線安裝解決無法安裝Proxy?AI插件問題(最新推薦)

    許多開發(fā)者嘗試通過安裝Proxy?AI等插件將AI能力引入IDEA,但在實(shí)際使用中常遭遇插件安裝失敗、網(wǎng)絡(luò)連接不穩(wěn)定或兼容性沖突等問題,本文給大家介紹IDEA集成DeepSeek通過離線安裝解決無法安裝Proxy?AI插件問題,感興趣的朋友一起看看吧
    2019-12-12
  • Java中的transient關(guān)鍵字解析

    Java中的transient關(guān)鍵字解析

    這篇文章主要介紹了Java中的 transient關(guān)鍵字解析,有時(shí)候我們的一些敏感信息比如密碼并不想序列化傳輸給對(duì)方,這個(gè)時(shí)候transient關(guān)鍵字就派上用場(chǎng)了,如果一個(gè)類的變量加上了transient關(guān)鍵字那么這個(gè)字段就不會(huì)被序列化,需要的朋友可以參考下
    2023-09-09
  • SpringBoot redis分布式緩存實(shí)現(xiàn)過程解析

    SpringBoot redis分布式緩存實(shí)現(xiàn)過程解析

    這篇文章主要介紹了SpringBoot redis分布式緩存實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-10-10
  • java加密MD5實(shí)現(xiàn)及密碼驗(yàn)證代碼實(shí)例

    java加密MD5實(shí)現(xiàn)及密碼驗(yàn)證代碼實(shí)例

    這篇文章主要介紹了java加密MD5實(shí)現(xiàn)及密碼驗(yàn)證代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-12-12
  • 微信開發(fā)之使用java獲取簽名signature

    微信開發(fā)之使用java獲取簽名signature

    這篇文章主要為大家詳細(xì)介紹了微信開發(fā)之使用java獲取簽名signature,感興趣的小伙伴們可以參考一下
    2016-08-08
  • java?ResourceBundle讀取properties文件方式

    java?ResourceBundle讀取properties文件方式

    這篇文章主要介紹了java?ResourceBundle讀取properties文件方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • springboot 使用yml配置文件自定義屬性的操作代碼

    springboot 使用yml配置文件自定義屬性的操作代碼

    在SpringBoot中yml/yaml文件可以自定義一些屬性,以供注入給自定義bean對(duì)象的屬性,主要通過空格和層次來實(shí)現(xiàn),類似于python代碼,本文通過實(shí)例代碼給大家介紹springboot 使用yml配置文件自定義屬性,感興趣的朋友跟隨小編一起看看吧
    2024-03-03
  • IDEA 自動(dòng)生成 JPA 實(shí)體類的圖文教程

    IDEA 自動(dòng)生成 JPA 實(shí)體類的圖文教程

    這篇文章主要介紹了IDEA 自動(dòng)生成 JPA 實(shí)體類的圖文教程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-07-07

最新評(píng)論