摘要:前言越來越多的項(xiàng)目開始嘗試寫單元測(cè)試,關(guān)于單元測(cè)試的好處以及原理已經(jīng)有很多資料了,這里不在做過多的講述,本文主要介紹單元測(cè)試在模塊化應(yīng)用中的一些思考,以及如何優(yōu)雅的寫單元測(cè)試。最后,依賴注入來寫單元測(cè)試。
本文由作者潘威授權(quán)網(wǎng)易云社區(qū)發(fā)布。
前言
越來越多的項(xiàng)目開始嘗試寫單元測(cè)試,關(guān)于單元測(cè)試的好處以及原理已經(jīng)有很多資料了,這里不在做過多的講述,本文主要介紹單元測(cè)試在模塊化應(yīng)用中的一些思考,以及如何優(yōu)雅的寫單元測(cè)試。
易于測(cè)試的代碼
單元測(cè)試最大的痛點(diǎn)就是代碼耦合,比如直接持有第三方庫的引用、不合理的跨層調(diào)用等等,除此之外,static method、new object、singleton 都是不利于測(cè)試的代碼方式, 這就意味著需要 mock 大量的替身類,增加了測(cè)試成本,應(yīng)該盡量避免,同時(shí)使用依賴注入的方式來代替。
如何做好單元測(cè)試?
首先,在模塊化應(yīng)用中應(yīng)該創(chuàng)建公共的單元測(cè)試模塊,里面可以放一些公共的 BaseTest、Shadow Class、Utils、Junit rules 等等,在業(yè)務(wù)模塊中直接 dependency 進(jìn)來即可,提高寫單元測(cè)試的效率。
其次,明確需要測(cè)試的代碼。剛開始的時(shí)候,可以只測(cè)中間邏輯層和工具類,這部分代碼相對(duì)“干凈”,易于測(cè)試,也是邏輯分支最集中的地方。
最后,依賴注入來寫單元測(cè)試。試想一下 mock 的類都能夠自動(dòng)完成注入,是不是很爽?這樣能大大提高編寫測(cè)試用例的速度,避免重復(fù)的 mock 替身類和靜態(tài)方法,并提高測(cè)試代碼的可讀性。
所以,我們引入了DI框架來做這件事情!
1、開發(fā)階段
我們只需要在一個(gè)類似于 dependency 工廠的地方統(tǒng)一生產(chǎn)這些 dependency 對(duì)象,以及這些 dependency 的 dependency。所有需要用到這些 dependency 的地方都從這個(gè)工廠里面去獲取。
2、測(cè)試階段
定義一個(gè)同樣的 dependency 工廠,不同的是,該工廠生產(chǎn)的是測(cè)試所需要的 Shadow 替身,能夠自動(dòng)識(shí)別依賴關(guān)系,并實(shí)現(xiàn)自動(dòng)注入!
Dagger2 的應(yīng)用
沒錯(cuò)!前面提到的 DI 框架就是 Dagger2,為了降低風(fēng)險(xiǎn)并減少使用成本,選擇了一個(gè)模塊進(jìn)行嘗試,Dagger2 既能實(shí)現(xiàn)模塊內(nèi)的自動(dòng)注入,又能向外提供注入能力,實(shí)現(xiàn)跨模塊的注入。
在 Dagger2 里,生產(chǎn)這些 dependency 的工廠叫做 Module ,然而使用者并不是直接向 Module 要 dependency,而是有一個(gè)專門的“工廠管理員”,負(fù)責(zé)接收使用者的要求,然后到 Module 里面去找到相應(yīng)的 dependency 對(duì)象,最后提供給使用者。這個(gè)“工廠管理員”叫做 Component?;旧希@就是 Dagger2 里面最重要的兩個(gè)概念。
上圖是 Dagger2 在模塊之間的依賴關(guān)系,本文只介紹模塊內(nèi)的應(yīng)用以及單元測(cè)試的實(shí)現(xiàn)。
1、創(chuàng)建模塊級(jí)的 LibComponent 和 LibModule
LibModule里面定義了整個(gè)模塊都要用的dependency,比如PersonalContentInstance 、Scope、 DataSource等等,所以DaggerLibComponent的存在是唯一的,在模塊初始化的時(shí)候創(chuàng)建好,放在一個(gè)地方便于獲取。
mInstance.mComponent = DaggerPersonalContentLibComponent.builder()
.personnalContentLibModule(new PersonnalContentLibModule()) .build();
2、創(chuàng)建 Frame 級(jí)別的 FrameComponent 和 FrameModule
FrameModule 里面定義了某個(gè)頁面用到的 dependency,比如 Context、Handler、Logic、Adapter 等等,每個(gè)頁面對(duì)應(yīng)一個(gè) DaggerFrameComponent,在頁面的 onCreate() 里面創(chuàng)建好。
3、FrameComponent 依賴于 LibComponent
在 Frame 中可以享受到 LibComponent 中全局依賴的注入,只需要在頁面初始化的時(shí)候完成注入即可。
DaggerFrameComponent.builder()
.libComponent(mInstance.getComponent()) .frameModule(new FrameModule(this)) .build() .injectMembers(this);
再看看單元測(cè)試?yán)锩嫒绾蝸韒ock dependency? 比如,LearnRecordDetailLogic 會(huì)調(diào)用mScope 和 mDataSource 中的方法,而 IPersonalContentScope 和 IDataSource 的實(shí)例對(duì)象是從 Dagger2 的 Component 里面獲取的,怎樣把 mScope 和 mDataSource 給 mock 掉呢?
實(shí)際上,LearnRecordDetailLogic 向 DaggerLibComponent 獲取實(shí)例調(diào)用的是 PersonnalContentLibModule 中的 provideDataSource() 和 provideScope() 方法,最后返回給 LearnRecordDetailLogic ,也就是說,真正實(shí)例化 IPersonalContentScope 和 IDataSource 的地方是在 PersonnalContentLibModule。
@Modulepublic class PersonnalContentLibModule {
...... @PerLibrary @Provides PersonalContentInstance providePersonalContentInstance() { return PersonalContentInstance.getInstance(); } @PerLibrary @Provides IPersonalContentScope provideScope(PersonalContentInstance instance) { return instance.getScope(); } @PerLibrary @Provides IDataSource provideDataSource(PersonalContentInstance instance) { return instance.getDataSourse(); }
}
前面創(chuàng)建 DaggerLibComponent 的時(shí)候,給它的 builder 傳遞了一個(gè) PersonnalContentLibModule 對(duì)象,如果我們傳給 DaggerLibComponent 的 Module 是一個(gè) TestModule,在它的 provide 方法被調(diào)用時(shí),返回一個(gè) mock 的 IPersonalContentScope 和 IDataSource,那么在測(cè)試代碼中獲得的,不就是 mock 后的替身對(duì)象嗎?
public class PersonnalContentLibTestModule extends PersonnalContentLibModule {
...... @Override PersonalContentInstance providePersonalContentInstance() { return PowerMockito.mock(PersonalContentInstance.class); } @Override IPersonalContentScope provideScope(PersonalContentInstance instance) { return PowerMockito.mock(IPersonalContentScope.class); } @Override IDataSource provideDataSource(PersonalContentInstance instance) { return PowerMockito.mock(IDataSource.class); }
}
以上就是 Dagger2 在單元測(cè)試?yán)锏膽?yīng)用。在 LibModule 的基礎(chǔ)上派生出一個(gè) LibTestModule,除此之外,LearnRecordDetailLogic 還用到了 Context 和 Handler 對(duì)象,所以需要?jiǎng)?chuàng)建一個(gè)Frame級(jí)別的 Module,然后 override 掉 provide方法,讓它返回你想要的 mock 對(duì)象。
看一下效果,越復(fù)雜的類越能發(fā)揮出 Dagger2 的威力!
//使用dagger之前mContext = mock(Context.class);
mHandler = mock(Handler.class);
mDataSource = mock(IDataSource.class);
mScope = mock(IPersonalContentScope.class);
mContentInstance = mock(PersonalContentInstance.class);
when(mContentInstance.getDataSourse()).thenReturn(mDataSource);
when(mContentInstance.getScope()).thenReturn(mScope);
mockStatic(PersonalContentInstance.class);
when(PersonalContentInstance.getInstance()).thenReturn(mContentInstance);//daggerDaggerFrameTestComponent.builder()
.libComponent(ComponentUtil.getLibTestComponent) .frameTestModule(new FrameTestModule()) .build() .inject(this);
總結(jié)
本文介紹了 Dagger2 在模塊內(nèi)以及單元測(cè)試中的應(yīng)用,DI是一種很好的開發(fā)模式,即使不做單元測(cè)試,也會(huì)讓我們的代碼更加簡潔、干凈、解耦,只不過在單元測(cè)試中發(fā)揮出了更大的威力,讓很多難測(cè)的代碼測(cè)試起來更加容易。
最后,介紹一下 Dagger2 的配置方法:
在模塊的 build.gradle 中添加
dependencies {
//other dependencies //Dagger2 compile "com.google.dagger:dagger:${DAGGER_VERSION}" annotationProcessor "com.google.dagger:dagger-compiler:${DAGGER_VERSION}"}
正常情況下,main 目錄下的源代碼 build 后,生成代碼放在 /build/generated/source/apt/buildType 下面,但是 test 目錄下的測(cè)試代碼,在 compile-time 階段卻無法識(shí)別。查看 build 目錄,發(fā)現(xiàn)存在這部分代碼,但是無法正常 import 進(jìn)來。所以還需要在 build.gradle 中添加如下代碼:
android.libraryVariants.all { def aptOutputDir = new File(buildDir, "generated/source/apt/${it.unitTestVariant.dirName}")
it.unitTestVariant.addJavaSourceFoldersToModel(aptOutputDir)
}
免費(fèi)領(lǐng)取驗(yàn)證碼、內(nèi)容安全、短信發(fā)送、直播點(diǎn)播體驗(yàn)包及云服務(wù)器等套餐
更多網(wǎng)易技術(shù)、產(chǎn)品、運(yùn)營經(jīng)驗(yàn)分享請(qǐng)?jiān)L問網(wǎng)易云社區(qū)。
文章來源: 網(wǎng)易云社區(qū)
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/25359.html
摘要:必然的,他們會(huì)拋棄標(biāo)準(zhǔn)庫中的,使用或者發(fā)明自己心儀的單元測(cè)試框架。究其原因,一些人會(huì)說時(shí)間寫代碼都不夠,哪還有空寫單元測(cè)試。最后我的個(gè)人觀點(diǎn),單元測(cè)試其實(shí)還有一個(gè)非常重要的作用,就是替代函數(shù)文檔注釋。希望從今天起,你的代碼也都有單元測(cè)試。 單元測(cè)試是每種編程語言必學(xué)的課題,是保護(hù)開發(fā)者的強(qiáng)力護(hù)盾,每個(gè)程序員都在時(shí)間允許的情況下盡可能多的寫單元測(cè)試,今天我們不討論其必要性,只拋磚引玉聊一...
摘要:實(shí)際開發(fā)中,如果每個(gè)包都去走一遍這些步驟,步驟好像確實(shí)有點(diǎn)多。 歡迎大家前往騰訊云+社區(qū),獲取更多騰訊海量技術(shù)實(shí)踐干貨哦~ 本文由小明plus發(fā)表 很多時(shí)候,我們可能想要用 typescript 語言來創(chuàng)建一些模塊,并提交到 npm 供別人使用, 那么在 2018 年,如果我想要初始化這樣的一個(gè)模塊,我需要做哪些步驟呢?: 答案是:創(chuàng)建一個(gè)優(yōu)雅的,對(duì)開發(fā)者友好的模塊,至少需要以下 15...
摘要:我寫過一些開源項(xiàng)目,在開源方面有一些經(jīng)驗(yàn),最近開到了阮老師的微博,深有感觸,現(xiàn)在一個(gè)開源項(xiàng)目涉及的東西確實(shí)挺多的,特別是對(duì)于新手來說非常不友好最近我寫了一個(gè),旨在從多方面快速幫大家搭建一個(gè)標(biāo)準(zhǔn)的庫,本文將已為例,介紹寫一個(gè)開源庫的知識(shí) 我寫過一些開源項(xiàng)目,在開源方面有一些經(jīng)驗(yàn),最近開到了阮老師的微博,深有感觸,現(xiàn)在一個(gè)開源項(xiàng)目涉及的東西確實(shí)挺多的,特別是對(duì)于新手來說非常不友好 show...
摘要:概述是一個(gè)輕量級(jí)的單元測(cè)試工具,基于二次開發(fā),使用它基于注解的方式,快速在本地進(jìn)行單元壓測(cè)并提供詳細(xì)的報(bào)告。當(dāng)和都有指定時(shí),以執(zhí)行次數(shù)多的為準(zhǔn)。測(cè)試報(bào)告最終的測(cè)試報(bào)告位于,使用瀏覽器打開即可。 概述 ContiPerf 是一個(gè)輕量級(jí)的單元測(cè)試工具,基于JUnit 4二次開發(fā),使用它基于注解的方式,快速在本地進(jìn)行單元壓測(cè)并提供詳細(xì)的報(bào)告。 Example 1. 新建 SpringBoot...
閱讀 3844·2023-04-25 16:32
閱讀 2225·2021-09-28 09:36
閱讀 2043·2021-09-06 15:02
閱讀 683·2021-09-02 15:21
閱讀 930·2019-08-30 15:56
閱讀 3527·2019-08-30 15:45
閱讀 1720·2019-08-30 13:09
閱讀 391·2019-08-29 16:05