中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Android中如何實現單元測試

發布時間:2021-07-12 11:52:07 來源:億速云 閱讀:133 作者:Leah 欄目:移動開發

這篇文章給大家介紹Android中如何實現單元測試,內容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。

本地jvm的單元測試

這種方式運行速度快,對運行環境沒有特殊要求,可以很方便的做自動化測試,是單元測試首選的方法

Instrumentation測試

Instrumentation測試需要運行在Android環境下,可以是模擬器或者手機等真實設備。這種方式運行速度慢,且嚴重依賴Android運行環境,更適合用來做集成測試

準備

我準備了一個簡單的APP,模擬一個耗時的網絡請求獲得一段數據并顯示在界面上,針對這個APP編寫單元測試用例并進行本地單元測試。

Android中如何實現單元測試

App運行效果

依賴庫

依賴庫作用
JUnit-4.12基礎得單元測試框架
Robolectric-3.8Android SDK測試框架
PowerMock-1.6.6模擬被測對象依賴的靜態方法
Mockito-1.10.19模擬被測對象依賴的對象

配置build.gradle

增加編譯選項,在測試中包含資源文件

 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主要是測試它各個生命周期的狀態變化、對外界輸入的響應是否符合預期,Activity測試完全依賴Android SDK,需要用Robolectric。

Robolectric是一個開源的單元測試框架,能夠完全模擬Android SDK并在JVM中運行。

UI依賴于Persenter,在Activity中通過靜態工廠方法創建依賴的Presenter實例,需要使用PowerMock來模擬創建Presenter過程,完成Presenter模擬對象的注入

配置

  • 通過@RunWith指定使用RobolectricTestRunner

  • 通過@Config配置Robolectric的運行環境

  • 通過@PrepareForTest配置PowerMock需要模擬的靜態類型

@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來構建并管理activity的生命周期,運行至onCreate階段,然后驗證這個階段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完全顯示以后,驗證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完全顯示以后,驗證button2的click操作是否調用了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依賴于底層的領域服務,也依賴上層View,demo中對領域服務的依賴沒有通過構造函數的方式注入,而是通過靜態工廠方法構建,還是需要用到PowerMock

配置

  1. 通過@RunWith指定使用PowerMockRunner

  2. 通過@PrepareForTest配置PowerMock需要模擬的靜態類型

@RunWith(PowerMockRunner.class)
@PrepareForTest({ServiceFactory.class})
 @Before
 public void setUp() {
  PowerMockito.mockStatic(ServiceFactory.class);
 }

成功路徑用例

驗證View的方法是否成功調用且調用參數是否一致

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

}

自動化測試

自動化測試一般是在持續集成環境中使用命令來執行單元測試

gradlew :app:testDebugUnitTest

關于Android中如何實現單元測試就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

文安县| 曲麻莱县| 广河县| 九寨沟县| 卓资县| 九江市| 五指山市| 廊坊市| 如皋市| 乌海市| 潍坊市| 淮安市| 舒兰市| 区。| 芦山县| 安丘市| 天台县| 清原| 莒南县| 天祝| 琼中| 延津县| 右玉县| 清新县| 西城区| 南靖县| 洱源县| 昂仁县| 无极县| 通榆县| 东兴市| 田阳县| 长宁县| 贵阳市| 紫金县| 玉林市| 获嘉县| 定南县| 桦甸市| 孝义市| 德令哈市|