欧美成人午夜免费全部完,亚洲午夜福利精品久久,а√最新版在线天堂,另类亚洲综合区图片小说区,亚洲欧美日韩精品色xxx

扣丁學(xué)堂Android培訓(xùn)解析如何對(duì)Android原生應(yīng)用進(jìn)行單元測(cè)試

2018-08-15 10:42:08 1492瀏覽

今天扣丁學(xué)堂Android培訓(xùn)老師給大家介紹一下關(guān)于關(guān)于在Android原生應(yīng)用開(kāi)發(fā)中,存在兩種單元測(cè)試:本地JVM測(cè)試和Instrumentation測(cè)試,那么下面我們來(lái)看一下關(guān)于如何對(duì)Android原生應(yīng)用進(jìn)行單元測(cè)試及示例代碼。

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

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

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



依賴(lài)庫(kù)



配置build.gradle

增加編譯選項(xiàng),在測(cè)試中包含資源文件

testOptions {
 unitTests {
  includeAndroidResources true
 }
}

添加測(cè)試依賴(lài)庫(kù)

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è)試完全依賴(lài)AndroidSDK,需要用Robolectric。

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

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

配置

通過(guò)@RunWith指定使用RobolectricTestRunner

通過(guò)@Config配置Robolectric的運(yùn)行環(huán)境

通過(guò)@PrepareForTest配置PowerMock需要模擬的靜態(tài)類(lè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用例

通過(guò)Robolectric的ActivityController來(lái)構(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());
}

ClickButton1用例

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());
}

ClickButton2用例

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è)試一般可以不用依賴(lài)AndroidSDK了,Presenter依賴(lài)于底層的領(lǐng)域服務(wù),也依賴(lài)上層View,demo中對(duì)領(lǐng)域服務(wù)的依賴(lài)沒(méi)有通過(guò)構(gòu)造函數(shù)的方式注入,而是通過(guò)靜態(tài)工廠方法構(gòu)建,還是需要用到PowerMock

配置

通過(guò)@RunWith指定使用PowerMockRunner

通過(guò)@PrepareForTest配置PowerMock需要模擬的靜態(tài)類(lè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ì)上層有依賴(lài),可以直接使用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è)試一般是在持續(xù)集成環(huán)境中使用命令來(lái)執(zhí)行單元測(cè)試:gradlew:app:testDebugUnitTest

以上就是關(guān)于扣丁學(xué)堂Android開(kāi)發(fā)培訓(xùn)單元測(cè)試最佳實(shí)踐的詳細(xì)介紹,希望對(duì)同學(xué)們學(xué)習(xí)有所幫助,想要了解更多詳情請(qǐng)登錄扣丁學(xué)堂官網(wǎng)咨詢或者關(guān)注微信公眾號(hào),里面有最新的扣丁學(xué)堂Android視頻教程等你來(lái)看!

扣丁學(xué)堂微信公眾號(hào)



【關(guān)注微信公眾號(hào)獲取更多學(xué)習(xí)資料】



查看更多關(guān)于“Android開(kāi)發(fā)技術(shù)的相關(guān)資訊>>

標(biāo)簽: Android培訓(xùn) Android學(xué)習(xí)路線 Android視頻教程 Android開(kāi)發(fā)培訓(xùn) Android培訓(xùn)班

熱門(mén)專(zhuān)區(qū)

暫無(wú)熱門(mén)資訊

課程推薦

微信
微博
15311698296

全國(guó)免費(fèi)咨詢熱線

郵箱:codingke@1000phone.com

官方群:148715490

北京千鋒互聯(lián)科技有限公司版權(quán)所有   北京市海淀區(qū)寶盛北里西區(qū)28號(hào)中關(guān)村智誠(chéng)科創(chuàng)大廈4層
京ICP備2021002079號(hào)-2   Copyright ? 2017 - 2022
返回頂部 返回頂部