詳解Android單元測試最佳實(shí)踐
目的
充分的單元測試就是提高代碼質(zhì)量最有效的手段之一,而單元測試嚴(yán)重依賴代碼的可測試性,本文主要通過一個簡單的DEMO演示如何對Android原生應(yīng)用進(jìn)行單元測試,同時示例代碼采用MVP模式以提高代碼的可讀性和可測試性
簡介
在Android原生應(yīng)用開發(fā)中,存在兩種單元測試:本地JVM測試和Instrumentation測試。本文僅介紹本地JVM測試
本地jvm的單元測試
這種方式運(yùn)行速度快,對運(yùn)行環(huán)境沒有特殊要求,可以很方便的做自動化測試,是單元測試首選的方法
Instrumentation測試
Instrumentation測試需要運(yùn)行在Android環(huán)境下,可以是模擬器或者手機(jī)等真實(shí)設(shè)備。這種方式運(yùn)行速度慢,且嚴(yán)重依賴Android運(yùn)行環(huán)境,更適合用來做集成測試
準(zhǔn)備
我準(zhǔn)備了一個簡單的APP,模擬一個耗時的網(wǎng)絡(luò)請求獲得一段數(shù)據(jù)并顯示在界面上,針對這個APP編寫單元測試用例并進(jìn)行本地單元測試。

App運(yùn)行效果
依賴庫
| 依賴庫 | 作用 |
|---|---|
| JUnit-4.12 | 基礎(chǔ)得單元測試框架 |
| Robolectric-3.8 | Android SDK測試框架 |
| PowerMock-1.6.6 | 模擬被測對象依賴的靜態(tài)方法 |
| Mockito-1.10.19 | 模擬被測對象依賴的對象 |
配置build.gradle
增加編譯選項(xiàng),在測試中包含資源文件
testOptions {
unitTests {
includeAndroidResources true
}
}
添加測試依賴庫
testImplementation 'junit:junit:4.12' testImplementation 'org.robolectric:robolectric:3.8' testImplementation 'org.robolectric:shadows-supportv4:3.8' testImplementation 'org.powermock:powermock-module-junit4:1.6.6' testImplementation 'org.powermock:powermock-module-junit4-rule:1.6.6' testImplementation 'org.powermock:powermock-api-mockito:1.6.6' testImplementation 'org.powermock:powermock-classloading-xstream:1.6.6' testImplementation 'org.mockito:mockito-all:1.10.19'
測試Activity
測試Activity主要是測試它各個生命周期的狀態(tài)變化、對外界輸入的響應(yīng)是否符合預(yù)期,Activity測試完全依賴Android SDK,需要用Robolectric。
Robolectric是一個開源的單元測試框架,能夠完全模擬Android SDK并在JVM中運(yùn)行。
UI依賴于Persenter,在Activity中通過靜態(tài)工廠方法創(chuàng)建依賴的Presenter實(shí)例,需要使用PowerMock來模擬創(chuàng)建Presenter過程,完成Presenter模擬對象的注入
配置
- 通過@RunWith指定使用RobolectricTestRunner
- 通過@Config配置Robolectric的運(yùn)行環(huán)境
- 通過@PrepareForTest配置PowerMock需要模擬的靜態(tài)類型
@RunWith(RobolectricTestRunner.class)
@Config(sdk = 21, constants = BuildConfig.class)
@PowerMockIgnore({"org.mockito.*", "org.robolectric.*", "android.*"})
@PrepareForTest({PresenterFactory.class})
@Before
public void setUp() {
appContext = RuntimeEnvironment.application.getApplicationContext();
PowerMockito.mockStatic(PresenterFactory.class);
}
onCreate用例
通過Robolectric的ActivityController來構(gòu)建并管理activity的生命周期,運(yùn)行至onCreate階段,然后驗(yàn)證這個階段text1是否正確初始化
@Test
public void onCreate_text1() {
MainActivity activity = Robolectric.buildActivity(MainActivity.class).create().get();
String expect = appContext.getString(R.string.hell_world);
assertEquals(expect, ((TextView)activity.findViewById(R.id.lbl_text1)).getText());
}
Click Button1用例
Activity完全顯示以后,驗(yàn)證button1的click操作是否顯示toast消息
@Test
public void btn1_click() {
MainActivity activity = Robolectric.setupActivity(MainActivity.class);
activity.findViewById(R.id.btn_1).performClick();
String expect = appContext.getString(R.string.hell_world);
assertEquals(expect, ShadowToast.getTextOfLatestToast());
}
Click Button2用例
Activity完全顯示以后,驗(yàn)證button2的click操作是否調(diào)用了presenter的fetch方法
@Test
public void btn2_click() {
MainContract.Presenter presenter = Mockito.mock(MainContract.Presenter.class);
PowerMockito.when(PresenterFactory.create(Mockito.any(MainContract.View.class), Mockito.any(AppExecutors.class)))
.thenReturn(presenter);
MainActivity activity = Robolectric.setupActivity(MainActivity.class);
activity.findViewById(R.id.btn_2).performClick();
Mockito.verify(presenter, Mockito.times(1))
.fetch();
}
測試Presenter
Presenter的測試一般可以不用依賴Android SDK了,Presenter依賴于底層的領(lǐng)域服務(wù),也依賴上層View,demo中對領(lǐng)域服務(wù)的依賴沒有通過構(gòu)造函數(shù)的方式注入,而是通過靜態(tài)工廠方法構(gòu)建,還是需要用到PowerMock
配置
- 通過@RunWith指定使用PowerMockRunner
- 通過@PrepareForTest配置PowerMock需要模擬的靜態(tài)類型
@RunWith(PowerMockRunner.class)
@PrepareForTest({ServiceFactory.class})
@Before
public void setUp() {
PowerMockito.mockStatic(ServiceFactory.class);
}
成功路徑用例
驗(yàn)證View的方法是否成功調(diào)用且調(diào)用參數(shù)是否一致
@Test
public void fetch_success() {
String expected = "hello world";
SlowService service = Mockito.mock(SlowService.class);
Mockito.when(service.fetch()).thenReturn(expected);
PowerMockito.when(ServiceFactory.create())
.thenReturn(service);
MainContract.View view = Mockito.mock(MainContract.View.class);
MainPresenter presenter = new MainPresenter(view, executors);
presenter.fetch();
Mockito.verify(service, Mockito.times(1)).fetch();
Mockito.verify(view, Mockito.times(1)).onFetchStarted();
Mockito.verify(view, Mockito.times(1)).onFetchCompleted();
Mockito.verify(view, Mockito.times(0)).onFetchFailed(Mockito.anyObject());
ArgumentCaptor<String> captor = ArgumentCaptor.forClass(String.class);
Mockito.verify(view, Mockito.times(1)).onFetchSuccess(captor.capture());
assertEquals(expected, captor.getValue());
}
失敗路徑用例
@Test
public void fetch_failed() {
RuntimeException exception = new RuntimeException("fetch failed");
SlowService service = Mockito.mock(SlowService.class);
Mockito.when(service.fetch()).thenThrow(exception);
PowerMockito.when(ServiceFactory.create())
.thenReturn(service);
MainContract.View view = Mockito.mock(MainContract.View.class);
MainPresenter presenter = new MainPresenter(view, executors);
presenter.fetch();
Mockito.verify(service, Mockito.times(1)).fetch();
Mockito.verify(view, Mockito.times(1)).onFetchStarted();
Mockito.verify(view, Mockito.times(1)).onFetchCompleted();
ArgumentCaptor<Throwable> captor = ArgumentCaptor.forClass(Throwable.class);
Mockito.verify(view, Mockito.times(1)).onFetchFailed(captor.capture());
assertEquals(exception, captor.getValue());
Mockito.verify(view, Mockito.times(0)).onFetchSuccess(Mockito.anyString());
}
測試Service
Service不會對上層有依賴,可以直接使用JUnit測試
public class SlowServiceImplTest {
@Test
public void fetch_data() {
SlowServiceImpl impl = new SlowServiceImpl();
String data = impl.fetch();
assertEquals("from slow service", data);
}
}
自動化測試
自動化測試一般是在持續(xù)集成環(huán)境中使用命令來執(zhí)行單元測試
gradlew :app:testDebugUnitTest
總結(jié)
寫完這個demo,總覺得給Android APP做單元測試還是非常簡單的,作為一個優(yōu)秀的程序員,怎么能夠不關(guān)注自己的代碼質(zhì)量呢,還是自己動手試試吧
源碼下載
https://github.com/hziee514/android-testing
參考資料
Robolectric
Using PowerMock
Mockito
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 詳解appium+python 啟動一個app步驟
- Python腳本在Appium庫上對移動應(yīng)用實(shí)現(xiàn)自動化測試
- android開機(jī)自啟動APP及使用adb命令測試方法
- Android利用Espresso進(jìn)行UI自動化測試的方法詳解
- 在Android打包中區(qū)分測試和正式環(huán)境淺析
- Android單元測試之對Activity的測試示例
- 淺談Android單元測試的作用以及簡單示例
- Android和iOS 測試五個最好的開源自動化工具
- Android 中構(gòu)建快速可靠的 UI 測試
- 簡單談?wù)刟ndroid studio 的單元測試
- Android Monkey壓力測試詳細(xì)介紹
- Ubuntu中為Android系統(tǒng)上實(shí)現(xiàn)內(nèi)置C可執(zhí)行程序測試Linux內(nèi)核驅(qū)動程序
- Android App開發(fā)的自動化測試框架UI Automator使用教程
- Android自動測試工具M(jìn)onkey的實(shí)現(xiàn)方法
- Android測試中Appium的一些錯誤解決技巧
相關(guān)文章
Android編程實(shí)現(xiàn)向桌面添加快捷方式的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)向桌面添加快捷方式的方法,結(jié)合實(shí)例形式詳細(xì)分析了Android添加桌面快捷方式的操作技巧,需要的朋友可以參考下2016-11-11
Android定時器和倒計時實(shí)現(xiàn)淘寶秒殺功能
這篇文章主要為大家詳細(xì)介紹了Android定時器和倒計時實(shí)現(xiàn)淘寶秒殺功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-02-02
android 開發(fā)中使用okhttp上傳文件到服務(wù)器
在開發(fā)android手機(jī)客戶端,常常會需要上傳文件到服務(wù)器,使用okhttp會是一個很好的選擇,它使用很簡單,而且運(yùn)行效率也很高,下面小編給大家?guī)砹薬ndroid 開發(fā)中使用okhttp上傳文件到服務(wù)器功能,一起看看吧2018-01-01
安卓(Android)中如何實(shí)現(xiàn)滑動導(dǎo)航
導(dǎo)航是移動應(yīng)用最重要的方面之一,對用戶體驗(yàn)是良好還是糟糕起著至關(guān)重要的作用。好的導(dǎo)航可以讓一款應(yīng)用更加易用并且讓用戶快速上手。相反,糟糕的應(yīng)用導(dǎo)航很容易讓人討厭,并遭到用戶的拋棄。2014-08-08
Android nativePollOnce函數(shù)解析
這篇文章主要介紹了Android nativePollOnce函數(shù)解析的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-03-03
Android用PopupWindow實(shí)現(xiàn)新浪微博的分組信息實(shí)例
PopupWindow可以實(shí)現(xiàn)浮層效果,而且可以自定義顯示位置,本篇文章主要介紹Android用PopupWindow實(shí)現(xiàn)新浪微博的分組信息,有需要的可以了解一下。2016-11-11

