Android單元測(cè)試最佳實(shí)踐

目的

充分的單元測(cè)試就是提高代碼質(zhì)量最有效的手段之一切油,而單元測(cè)試嚴(yán)重依賴代碼的可測(cè)試性,本文主要通過一個(gè)簡(jiǎn)單的DEMO演示如何對(duì)Android原生應(yīng)用進(jìn)行單元測(cè)試件相,同時(shí)示例代碼采用MVP模式以提高代碼的可讀性和可測(cè)試性

簡(jiǎn)介

在Android原生應(yīng)用開發(fā)中,存在兩種單元測(cè)試:本地JVM測(cè)試和Instrumentation測(cè)試。本文僅介紹本地JVM測(cè)試

  • 本地jvm的單元測(cè)試
    這種方式運(yùn)行速度快疙赠,對(duì)運(yùn)行環(huán)境沒有特殊要求,可以很方便的做自動(dòng)化測(cè)試朦拖,是單元測(cè)試首選的方法

  • Instrumentation測(cè)試
    Instrumentation測(cè)試需要運(yùn)行在Android環(huán)境下圃阳,可以是模擬器或者手機(jī)等真實(shí)設(shè)備。這種方式運(yùn)行速度慢璧帝,且嚴(yán)重依賴Android運(yùn)行環(huán)境捍岳,更適合用來做集成測(cè)試

準(zhǔn)備

我準(zhǔn)備了一個(gè)簡(jiǎn)單的APP,模擬一個(gè)耗時(shí)的網(wǎng)絡(luò)請(qǐng)求獲得一段數(shù)據(jù)并顯示在界面上睬隶,針對(duì)這個(gè)APP編寫單元測(cè)試用例并進(jìn)行本地單元測(cè)試锣夹。

App運(yùn)行效果

依賴庫

依賴庫 作用
JUnit-4.12 基礎(chǔ)得單元測(cè)試框架
Robolectric-3.8 Android SDK測(cè)試框架
PowerMock-1.6.6 模擬被測(cè)對(duì)象依賴的靜態(tài)方法
Mockito-1.10.19 模擬被測(cè)對(duì)象依賴的對(duì)象

配置build.gradle

  • 增加編譯選項(xiàng),在測(cè)試中包含資源文件
    testOptions {
        unitTests {
            includeAndroidResources true
        }
    }
  • 添加測(cè)試依賴庫
    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'

測(cè)試Activity

測(cè)試Activity主要是測(cè)試它各個(gè)生命周期的狀態(tài)變化苏潜、對(duì)外界輸入的響應(yīng)是否符合預(yù)期银萍,Activity測(cè)試完全依賴Android SDK,需要用Robolectric恤左。

Robolectric是一個(gè)開源的單元測(cè)試框架贴唇,能夠完全模擬Android SDK并在JVM中運(yùn)行。

UI依賴于Persenter飞袋,在Activity中通過靜態(tài)工廠方法創(chuàng)建依賴的Presenter實(shí)例戳气,需要使用PowerMock來模擬創(chuàng)建Presenter過程,完成Presenter模擬對(duì)象的注入

配置

  • 通過@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)證這個(gè)階段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();
    }

測(cè)試Presenter

Presenter的測(cè)試一般可以不用依賴Android SDK了呀袱,Presenter依賴于底層的領(lǐng)域服務(wù)芯肤,也依賴上層View,demo中對(duì)領(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());
    }

測(cè)試Service

Service不會(huì)對(duì)上層有依賴,可以直接使用JUnit測(cè)試

public class SlowServiceImplTest {

    @Test
    public void fetch_data() {
        SlowServiceImpl impl = new SlowServiceImpl();
        String data = impl.fetch();
        assertEquals("from slow service", data);
    }

}

自動(dòng)化測(cè)試

自動(dòng)化測(cè)試一般是在持續(xù)集成環(huán)境中使用命令來執(zhí)行單元測(cè)試

gradlew :app:testDebugUnitTest

總結(jié)

寫完這個(gè)demo油吭,總覺得給Android APP做單元測(cè)試還是非常簡(jiǎn)單的击蹲,作為一個(gè)優(yōu)秀的程序員,怎么能夠不關(guān)注自己的代碼質(zhì)量呢婉宰,還是自己動(dòng)手試試吧

源碼下載

https://github.com/hziee514/android-testing

參考資料

Robolectric
Using PowerMock
Mockito
PowerMock

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末歌豺,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子心包,更是在濱河造成了極大的恐慌类咧,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,376評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蟹腾,死亡現(xiàn)場(chǎng)離奇詭異痕惋,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)娃殖,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,126評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門值戳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人炉爆,你說我怎么就攤上這事堕虹。” “怎么了芬首?”我有些...
    開封第一講書人閱讀 156,966評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵赴捞,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我郁稍,道長(zhǎng)赦政,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,432評(píng)論 1 283
  • 正文 為了忘掉前任艺晴,我火速辦了婚禮昼钻,結(jié)果婚禮上掸屡,老公的妹妹穿的比我還像新娘封寞。我一直安慰自己菩彬,他們只是感情好餐茵,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,519評(píng)論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著安拟,像睡著了一般盏求。 火紅的嫁衣襯著肌膚如雪抖锥。 梳的紋絲不亂的頭發(fā)上亿眠,一...
    開封第一講書人閱讀 49,792評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音磅废,去河邊找鬼纳像。 笑死,一個(gè)胖子當(dāng)著我的面吹牛拯勉,可吹牛的內(nèi)容都是我干的竟趾。 我是一名探鬼主播,決...
    沈念sama閱讀 38,933評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼宫峦,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼岔帽!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起导绷,我...
    開封第一講書人閱讀 37,701評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤犀勒,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后妥曲,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體贾费,經(jīng)...
    沈念sama閱讀 44,143評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,488評(píng)論 2 327
  • 正文 我和宋清朗相戀三年檐盟,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了铸本。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,626評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡遵堵,死狀恐怖箱玷,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情陌宿,我是刑警寧澤锡足,帶...
    沈念sama閱讀 34,292評(píng)論 4 329
  • 正文 年R本政府宣布,位于F島的核電站壳坪,受9級(jí)特大地震影響舶得,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜爽蝴,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,896評(píng)論 3 313
  • 文/蒙蒙 一沐批、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蝎亚,春花似錦九孩、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,742評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春宪拥,著一層夾襖步出監(jiān)牢的瞬間仿野,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來泰國打工她君, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留脚作,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,324評(píng)論 2 360
  • 正文 我出身青樓缔刹,卻偏偏與公主長(zhǎng)得像鳖枕,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子桨螺,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,494評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容