亚洲国产日韩欧美一区二区三区,精品亚洲国产成人av在线,国产99视频精品免视看7,99国产精品久久久久久久成人热,欧美日韩亚洲国产综合乱

目錄
問(wèn)題根源分析
解決方案:依賴注入 (Dependency Injection)
1. 修改生產(chǎn)代碼以接受依賴
2. 在測(cè)試中注入Spy對(duì)象
3. 在生產(chǎn)代碼中注入真實(shí)對(duì)象
總結(jié)與注意事項(xiàng)
首頁(yè) Java java教程 解決Mockito Spy方法未生效:理解依賴注入與測(cè)試策略

解決Mockito Spy方法未生效:理解依賴注入與測(cè)試策略

Aug 04, 2025 pm 07:24 PM

解決Mockito Spy方法未生效:理解依賴注入與測(cè)試策略

本文深入探討Mockito Spy樁定方法未生效的常見問(wèn)題。當(dāng)生產(chǎn)代碼獨(dú)立創(chuàng)建實(shí)例而非使用測(cè)試中的Spy對(duì)象時(shí),樁定將失效。核心解決方案是采用依賴注入(DI),允許測(cè)試環(huán)境注入Spy對(duì)象,生產(chǎn)環(huán)境注入真實(shí)對(duì)象,從而確保樁定生效,提升代碼可測(cè)試性。文章將通過(guò)代碼示例詳細(xì)闡述這一實(shí)踐。

在使用Mockito進(jìn)行單元測(cè)試時(shí),spy功能允許我們部分模擬一個(gè)真實(shí)對(duì)象,即可以調(diào)用真實(shí)方法,也可以樁定(stub)特定方法的行為。然而,一個(gè)常見的誤區(qū)是,即使我們對(duì)一個(gè)類的實(shí)例進(jìn)行了spy并樁定了其方法,生產(chǎn)代碼卻可能仍然調(diào)用真實(shí)對(duì)象的默認(rèn)行為,而不是我們預(yù)期的樁定值。

問(wèn)題根源分析

問(wèn)題的核心在于,Mockito的spy和mock對(duì)象是為了作為被測(cè)試單元的“依賴”而存在的。當(dāng)你的生產(chǎn)代碼在內(nèi)部自行創(chuàng)建了GetOptionBidPrice的實(shí)例時(shí),它使用的是一個(gè)全新的、真實(shí)的GetOptionBidPrice對(duì)象,而不是你在測(cè)試中精心構(gòu)造并樁定的spyGetOptionBidPrice對(duì)象。

考慮以下生產(chǎn)代碼片段:

// 生產(chǎn)代碼片段
public class MyService {
    public double calculateValue() {
        // 在這里,getOptionBidPrice 是一個(gè)全新的實(shí)例
        GetOptionBidPrice getOptionBidPrice = new GetOptionBidPrice(...);
        double bidPrice = getOptionBidPrice.getBidPrice();
        // ... 使用 bidPrice 進(jìn)行后續(xù)計(jì)算
        return bidPrice * 2;
    }
}

以及對(duì)應(yīng)的測(cè)試代碼嘗試對(duì)GetOptionBidPrice進(jìn)行樁定:

// 測(cè)試代碼片段
@Test
void testCalculateValueWithStubbedBidPrice() {
    GetOptionBidPrice spyGetOptionBidPrice = spy(GetOptionBidPrice.class);
    // 樁定 getBidPrice 方法返回 100.0
    doReturn(100.0).when(spyGetOptionBidPrice).getBidPrice();

    // 問(wèn)題:MyService.calculateValue() 內(nèi)部仍會(huì)創(chuàng)建新的 GetOptionBidPrice 實(shí)例
    // 因此,上面樁定的 spyGetOptionBidPrice 并不會(huì)被 MyService 使用
    MyService myService = new MyService();
    double result = myService.calculateValue();

    // 預(yù)期 result 為 200.0 (100.0 * 2),但實(shí)際可能因?yàn)?getBidPrice 返回 0.0 而得到 0.0
    assertEquals(200.0, result);
}

在這種情況下,盡管你在測(cè)試中樁定了spyGetOptionBidPrice.getBidPrice(),但MyService.calculateValue()方法內(nèi)部自行創(chuàng)建的GetOptionBidPrice實(shí)例與這個(gè)spy對(duì)象毫無(wú)關(guān)聯(lián)。因此,getBidPrice()方法調(diào)用的是真實(shí)對(duì)象的默認(rèn)實(shí)現(xiàn),導(dǎo)致樁定失效。

解決方案:依賴注入 (Dependency Injection)

解決此問(wèn)題的關(guān)鍵在于采用依賴注入(Dependency Injection,簡(jiǎn)稱DI)。依賴注入是一種設(shè)計(jì)模式,它允許將對(duì)象的依賴關(guān)系從對(duì)象內(nèi)部移除,轉(zhuǎn)而由外部容器或框架在運(yùn)行時(shí)提供。這極大地提高了代碼的模塊化、可測(cè)試性和可維護(hù)性。

1. 修改生產(chǎn)代碼以接受依賴

不再在方法內(nèi)部創(chuàng)建依賴對(duì)象,而是將其作為方法的參數(shù)或類的構(gòu)造器參數(shù)傳入。

修改前 (問(wèn)題代碼):

public class MyService {
    public double calculateValue() {
        GetOptionBidPrice getOptionBidPrice = new GetOptionBidPrice(...); // 內(nèi)部創(chuàng)建依賴
        double bidPrice = getOptionBidPrice.getBidPrice();
        return bidPrice * 2;
    }
}

修改后 (使用依賴注入):

方法參數(shù)注入:

public class MyService {
    // 將 GetOptionBidPrice 作為方法參數(shù)傳入
    public double calculateValue(GetOptionBidPrice getOptionBidPrice) {
        double bidPrice = getOptionBidPrice.getBidPrice();
        return bidPrice * 2;
    }
}

構(gòu)造器注入 (推薦,更符合面向?qū)ο笤O(shè)計(jì)):

public class MyService {
    private final GetOptionBidPrice getOptionBidPrice;

    // 通過(guò)構(gòu)造器注入 GetOptionBidPrice 實(shí)例
    public MyService(GetOptionBidPrice getOptionBidPrice) {
        this.getOptionBidPrice = getOptionBidPrice;
    }

    public double calculateValue() {
        double bidPrice = getOptionBidPrice.getBidPrice();
        return bidPrice * 2;
    }
}

2. 在測(cè)試中注入Spy對(duì)象

通過(guò)依賴注入,我們現(xiàn)在可以在測(cè)試中將樁定好的spy對(duì)象傳入MyService,確保MyService內(nèi)部使用的是我們期望的模擬行為。

import org.junit.jupiter.api.Test;
import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.assertEquals;

class MyServiceTest {

    @Test
    void testCalculateValueWithStubbedBidPrice() {
        // 1. 創(chuàng)建并樁定 Spy 對(duì)象
        GetOptionBidPrice spyGetOptionBidPrice = spy(new GetOptionBidPrice(...)); // 如果 GetOptionBidPrice 有構(gòu)造器參數(shù),這里需要傳入
        doReturn(100.0).when(spyGetOptionBidPrice).getBidPrice();

        // 2. 實(shí)例化 MyService,并注入樁定好的 Spy 對(duì)象
        // 如果使用構(gòu)造器注入:
        MyService myService = new MyService(spyGetOptionBidPrice);
        // 如果使用方法參數(shù)注入,則在調(diào)用 calculateValue 時(shí)傳入:
        // MyService myService = new MyService(); // 假設(shè) MyService 還有無(wú)參構(gòu)造器

        // 3. 調(diào)用被測(cè)試方法
        double result = myService.calculateValue(); // 如果是方法參數(shù)注入,這里傳入 spyGetOptionBidPrice

        // 4. 驗(yàn)證結(jié)果
        assertEquals(200.0, result, "樁定值應(yīng)被正確使用并計(jì)算");

        // 5. 驗(yàn)證 spy 方法是否被調(diào)用 (可選)
        verify(spyGetOptionBidPrice, times(1)).getBidPrice();
    }
}

3. 在生產(chǎn)代碼中注入真實(shí)對(duì)象

在生產(chǎn)環(huán)境中,你將注入一個(gè)真實(shí)的GetOptionBidPrice實(shí)例。這通常通過(guò)Spring、Guice等DI框架自動(dòng)完成,或者手動(dòng)創(chuàng)建并傳遞。

// 生產(chǎn)代碼中的調(diào)用示例
public class ApplicationRunner {
    public static void main(String[] args) {
        // 創(chuàng)建真實(shí)的 GetOptionBidPrice 實(shí)例
        GetOptionBidPrice realGetOptionBidPrice = new GetOptionBidPrice(...);

        // 創(chuàng)建 MyService 實(shí)例,并注入真實(shí)的依賴
        MyService myService = new MyService(realGetOptionBidPrice);

        // 調(diào)用業(yè)務(wù)方法
        double finalValue = myService.calculateValue();
        System.out.println("Calculated Value: " + finalValue);
    }
}

總結(jié)與注意事項(xiàng)

  • 依賴注入是關(guān)鍵: 任何一個(gè)類或方法如果需要使用其他對(duì)象的功能,都應(yīng)該通過(guò)依賴注入的方式獲取這些對(duì)象,而不是在內(nèi)部自行創(chuàng)建。這是實(shí)現(xiàn)高內(nèi)聚、低耦合、高可測(cè)試性代碼的基礎(chǔ)。
  • Spy與Mock的區(qū)別: spy用于部分模擬真實(shí)對(duì)象,保留真實(shí)行為;mock則完全模擬,所有方法默認(rèn)不執(zhí)行真實(shí)邏輯。無(wú)論使用哪種,它們都是為了被注入到被測(cè)試單元中。
  • 構(gòu)造器注入 vs. Setter注入 vs. 方法參數(shù)注入:
    • 構(gòu)造器注入通常是首選,因?yàn)樗WC了對(duì)象在創(chuàng)建時(shí)就擁有所有必要的依賴,使得對(duì)象始終處于有效狀態(tài),并且依賴是不可變的。
    • Setter注入允許在對(duì)象創(chuàng)建后設(shè)置依賴,適用于可選依賴或循環(huán)依賴(不推薦)。
    • 方法參數(shù)注入適用于特定方法需要某個(gè)臨時(shí)依賴的場(chǎng)景,但如果依賴是類級(jí)別的,則構(gòu)造器注入更合適。
  • 測(cè)試驅(qū)動(dòng)開發(fā) (TDD) 的實(shí)踐: 遵循TDD原則,在編寫功能代碼之前先寫測(cè)試,這會(huì)自然而然地引導(dǎo)你設(shè)計(jì)出易于測(cè)試的、低耦合的代碼結(jié)構(gòu),其中依賴注入是不可或缺的一部分。

通過(guò)采納依賴注入模式,你不僅解決了Mockito Spy樁定不生效的問(wèn)題,更重要的是,你的代碼將變得更加健壯、靈活和易于維護(hù)。

以上是解決Mockito Spy方法未生效:理解依賴注入與測(cè)試策略的詳細(xì)內(nèi)容。更多資訊請(qǐng)關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

本網(wǎng)站聲明
本文內(nèi)容由網(wǎng)友自願(yuàn)投稿,版權(quán)歸原作者所有。本站不承擔(dān)相應(yīng)的法律責(zé)任。如發(fā)現(xiàn)涉嫌抄襲或侵權(quán)的內(nèi)容,請(qǐng)聯(lián)絡(luò)admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費(fèi)脫衣圖片

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅(qū)動(dòng)的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費(fèi)的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費(fèi)的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強(qiáng)大的PHP整合開發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺(jué)化網(wǎng)頁(yè)開發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級(jí)程式碼編輯軟體(SublimeText3)

熱門話題

Laravel 教程
1597
29
PHP教程
1488
72
現(xiàn)代爪哇的異步編程技術(shù) 現(xiàn)代爪哇的異步編程技術(shù) Jul 07, 2025 am 02:24 AM

Java支持異步編程的方式包括使用CompletableFuture、響應(yīng)式流(如ProjectReactor)以及Java19 中的虛擬線程。 1.CompletableFuture通過(guò)鍊式調(diào)用提升代碼可讀性和維護(hù)性,支持任務(wù)編排和異常處理;2.ProjectReactor提供Mono和Flux類型實(shí)現(xiàn)響應(yīng)式編程,具備背壓機(jī)制和豐富的操作符;3.虛擬線程減少並發(fā)成本,適用於I/O密集型任務(wù),與傳統(tǒng)平臺(tái)線程相比更輕量且易於擴(kuò)展。每種方式均有適用場(chǎng)景,應(yīng)根據(jù)需求選擇合適工具並避免混合模型以保持簡(jiǎn)潔性

在Java中使用枚舉的最佳實(shí)踐 在Java中使用枚舉的最佳實(shí)踐 Jul 07, 2025 am 02:35 AM

在Java中,枚舉(enum)適合表示固定常量集合,最佳實(shí)踐包括:1.用enum表示固定狀態(tài)或選項(xiàng),提升類型安全和可讀性;2.為枚舉添加屬性和方法以增強(qiáng)靈活性,如定義字段、構(gòu)造函數(shù)、輔助方法等;3.使用EnumMap和EnumSet提高性能和類型安全性,因其基於數(shù)組實(shí)現(xiàn)更高效;4.避免濫用enum,如動(dòng)態(tài)值、頻繁變更或複雜邏輯場(chǎng)景應(yīng)使用其他方式替代。正確使用enum能提升代碼質(zhì)量並減少錯(cuò)誤,但需注意其適用邊界。

了解Java Nio及其優(yōu)勢(shì) 了解Java Nio及其優(yōu)勢(shì) Jul 08, 2025 am 02:55 AM

JavaNIO是Java1.4引入的新型IOAPI,1)面向緩衝區(qū)和通道,2)包含Buffer、Channel和Selector核心組件,3)支持非阻塞模式,4)相比傳統(tǒng)IO更高效處理並發(fā)連接。其優(yōu)勢(shì)體現(xiàn)在:1)非阻塞IO減少線程開銷,2)Buffer提升數(shù)據(jù)傳輸效率,3)Selector實(shí)現(xiàn)多路復(fù)用,4)內(nèi)存映射加快文件讀寫。使用時(shí)需注意:1)Buffer的flip/clear操作易混淆,2)非阻塞下需手動(dòng)處理不完整數(shù)據(jù),3)Selector註冊(cè)需及時(shí)取消,4)NIO並非適用於所有場(chǎng)景。

Java Classloader在內(nèi)部如何工作 Java Classloader在內(nèi)部如何工作 Jul 06, 2025 am 02:53 AM

Java的類加載機(jī)制通過(guò)ClassLoader實(shí)現(xiàn),其核心工作流程分為加載、鏈接和初始化三個(gè)階段。加載階段由ClassLoader動(dòng)態(tài)讀取類的字節(jié)碼並創(chuàng)建Class對(duì)象;鏈接包括驗(yàn)證類的正確性、為靜態(tài)變量分配內(nèi)存及解析符號(hào)引用;初始化則執(zhí)行靜態(tài)代碼塊和靜態(tài)變量賦值。類加載採(cǎi)用雙親委派模型,優(yōu)先委託父類加載器查找類,依次嘗試Bootstrap、Extension和ApplicationClassLoader,確保核心類庫(kù)安全且避免重複加載。開發(fā)者可自定義ClassLoader,如URLClassL

Hashmap在Java內(nèi)部如何工作? Hashmap在Java內(nèi)部如何工作? Jul 15, 2025 am 03:10 AM

HashMap在Java中通過(guò)哈希表實(shí)現(xiàn)鍵值對(duì)存儲(chǔ),其核心在於快速定位數(shù)據(jù)位置。 1.首先使用鍵的hashCode()方法生成哈希值,並通過(guò)位運(yùn)算轉(zhuǎn)換為數(shù)組索引;2.不同對(duì)象可能產(chǎn)生相同哈希值,導(dǎo)致衝突,此時(shí)以鍊錶形式掛載節(jié)點(diǎn),JDK8後鍊錶過(guò)長(zhǎng)(默認(rèn)長(zhǎng)度8)則轉(zhuǎn)為紅黑樹提升效率;3.使用自定義類作鍵時(shí)必須重寫equals()和hashCode()方法;4.HashMap動(dòng)態(tài)擴(kuò)容,當(dāng)元素?cái)?shù)超過(guò)容量乘以負(fù)載因子(默認(rèn)0.75)時(shí),擴(kuò)容並重新哈希;5.HashMap非線程安全,多線程下應(yīng)使用Concu

有效使用爪哇枚舉和最佳實(shí)踐 有效使用爪哇枚舉和最佳實(shí)踐 Jul 07, 2025 am 02:43 AM

Java枚舉不僅表示常量,還可封裝行為、攜帶數(shù)據(jù)、實(shí)現(xiàn)接口。 1.枚舉是類,用於定義固定實(shí)例,如星期、狀態(tài),比字符串或整數(shù)更安全;2.可攜帶數(shù)據(jù)和方法,如通過(guò)構(gòu)造函數(shù)傳值並提供訪問(wèn)方法;3.可使用switch處理不同邏輯,結(jié)構(gòu)清晰;4.可實(shí)現(xiàn)接口或抽象方法,使不同枚舉值具有差異化行為;5.注意避免濫用、硬編碼比較、依賴ordinal值,合理命名與序列化。

如何在Java中正確處理異常? 如何在Java中正確處理異常? Jul 06, 2025 am 02:43 AM

處理Java中的異常關(guān)鍵在於捕獲得當(dāng)、處理明確、不掩蓋問(wèn)題。一要按需捕獲具體異常類型,避免籠統(tǒng)catch,優(yōu)先處理checkedexception,運(yùn)行時(shí)異常應(yīng)提前判斷;二要使用日誌框架記錄異常,根據(jù)類型決定重試、回滾或拋出;三要利用finally塊釋放資源,推薦try-with-resources;四要合理定義自定義異常,繼承RuntimeException或Exception,攜帶上下文信息便於調(diào)試。

Java中的單例設(shè)計(jì)模式是什麼? Java中的單例設(shè)計(jì)模式是什麼? Jul 09, 2025 am 01:32 AM

單例設(shè)計(jì)模式在Java中通過(guò)私有構(gòu)造器和靜態(tài)方法確保一個(gè)類只有一個(gè)實(shí)例並提供全局訪問(wèn)點(diǎn),適用於控制共享資源的訪問(wèn)。實(shí)現(xiàn)方式包括:1.懶加載,即首次請(qǐng)求時(shí)才創(chuàng)建實(shí)例,適用於資源消耗大且不一定需要的情況;2.線程安全處理,通過(guò)同步方法或雙重檢查鎖定確保多線程環(huán)境下只創(chuàng)建一個(gè)實(shí)例,並減少性能影響;3.餓漢式加載,在類加載時(shí)直接初始化實(shí)例,適合輕量級(jí)對(duì)像或可接受提前初始化的場(chǎng)景;4.枚舉實(shí)現(xiàn),利用Java枚舉天然支持序列化、線程安全及防止反射攻擊的特性,是推薦的簡(jiǎn)潔可靠方式。不同實(shí)現(xiàn)方式可根據(jù)具體需求選

See all articles