Android利用Espresso進(jìn)行UI自動(dòng)化測試的方法詳解
為什么需要UI自動(dòng)化測試?
我有一個(gè)觀點(diǎn),對(duì)于重復(fù)的工作,那么程序都是可以代替的,我想這是作為一個(gè)程序員的一個(gè)基本素養(yǎng)(能偷懶的絕不干活)。UI自動(dòng)化測試就是為了應(yīng)付一些重復(fù)的工作,比如說測試某個(gè)功能,那么從應(yīng)用點(diǎn)擊,再經(jīng)過一系列的點(diǎn)擊頁面才能到達(dá)這個(gè)頁面,然后進(jìn)行測試,那么我們是不是可以寫段代碼讓app自動(dòng)跑起來,自動(dòng)來到那個(gè)界面進(jìn)行測試呢?答案是肯定的,這就是本文所要說的自動(dòng)化測試。
引言
谷歌2013年的時(shí)候開源了espress,谷歌的思路是,等到它足夠成熟和穩(wěn)定以后,將其遷移到Android SDK中,以此可見對(duì)他的重視。Google使用Espresso測試了他們自己的超過30個(gè)應(yīng)用程序,包括G+、Maps和Drive。
Espresso測試是非常容易實(shí)現(xiàn)的,由三步構(gòu)成:
- ViewMachers:尋找用來測試的View。
- ViewActions:發(fā)送交互事件。
- ViewAssertions:檢驗(yàn)測試結(jié)果
先看下官方給的示例,就能理解以上的三個(gè)步驟:
onView(withId(R.id.my_view)) // withId(R.id.my_view) is a ViewMatcher .perform(click()) // click() is a ViewAction .check(matches(isDisplayed())); // matches(isDisplayed()) is a ViewAssertion
Espresso框架是google官方大力推薦的一套測試框架,所以無論如何都要學(xué)習(xí)一下的.另外,自Android Studio2.2版本開始,google就為Espresso框架內(nèi)置了一個(gè)圖形化界面,用來自動(dòng)生成單元測試代碼。
接下來一起寫一demo測試,深入了解Espresso。
準(zhǔn)備
支持Espresso:
dependencies {
...
testCompile 'junit:junit:4.12'
androidTestCompile 'com.android.support.test.espresso:espresso-core:2.2.2', {
exclude group: 'com.android.support', module: 'support-annotations'
}
}
在dependencies中添加,一般默認(rèn)會(huì)有testCompile 'junit:junit:4.12',所以我們只需添加另一句即可。
defaultConfig{
...
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
在defaultConfig中添加如上語句,支持測試運(yùn)行。
創(chuàng)建Test類
特別注意,該類應(yīng)在androidTest文件夾下

- androidTest:進(jìn)行與Android相關(guān)(如調(diào)用Android設(shè)備等)測試;
- test:進(jìn)行簡單的只涉及java SE相關(guān)的測試。
舉個(gè)簡單例子:
@RunWith(AndroidJUnit4.class)
@LargeTest
public class MainActivityInstrumentationTest {
@Rule
public ActivityTestRule mActivityRule = new ActivityTestRule<>(
MainActivity.class);
@Test
public void sayHello(){
onView(withText("Say hello!")).perform(click());
onView(withId(R.id.textView)).check(matches(withText("Hello, World!")));
}
}
- 首先需要在測試用例類的類體前添加@RunWith的注解,并設(shè)置測試運(yùn)行平臺(tái)為AndroidJUnit4
- 如果允許測試需要較大消耗,可以使用@LargeTest注解
- 設(shè)置ActivityTestRule用來指明被測試的Activity,使用@Rule注解
- 測試方法必須以 test 開頭,并且使用@Test注解(否則會(huì)報(bào)找不到方法異常)
@Rule
@Rule public ActivityTestRule mTestRule = new ActivityTestRule<>(MainActivity.class);
這句話就定義了一個(gè)測試規(guī)則,可以看到構(gòu)造方法的參數(shù)里指定了一個(gè) MainActivity.class, 具體的體現(xiàn)就是當(dāng)你運(yùn)行這段測試代碼時(shí),app將會(huì)直接打開 MainActivity界面然后進(jìn)行你所定義的測試用例。 所以當(dāng)你想直接測試某個(gè)界面時(shí),你可以把那個(gè)界面填到這個(gè)參數(shù)里,這樣就直接打開你指定的界面進(jìn)行測試了。
@Test
@Test
public void testLogin() {
...
}
定義一個(gè)測試方法,當(dāng)你的測試類運(yùn)行時(shí),所執(zhí)行的代碼就是Test注解下的方法(Espresso還提供了其他的一些注解: 比如@After,@Before等,具體的用法可以去我上面寫的android官網(wǎng)上查看),當(dāng)然上面那段代碼對(duì)應(yīng)的就是testLogin測試方法,testLogin方法里所定義的就是要測試的內(nèi)容。
ViewMachers 查找View
使用onView方法找到view:其中參數(shù)可以是withId(通過資源id查找),withText(通過顯示內(nèi)容查找)有多個(gè)約束條件時(shí),可以使用allOf 如allOf(withText("Hello") ,withId(R.id.hello))
注意:
- 無論是通過withId()找控件還是通過
withText()找控件,或者其他方式比如withClassName(),withResourceName(),withTagKey()等方法,都要一定保證你所找的控件在當(dāng)前頁面確實(shí)存在且可見。 - 如果要測試AdapterView ,比如 ListView 或GridView等,使用上面的
onView()方法是無效的,因?yàn)锳dapterView的布局item是動(dòng)態(tài)呈現(xiàn)的,沒法直接指定,所以當(dāng)要測試AdapterView時(shí),請(qǐng)把onView()方法換成onData()方法,與onView()方法返回ViewInteraction類似,onData()方法返回DataInteraction,二者用法基本都是一樣的。
ViewActions 執(zhí)行事件
對(duì)View的操作:perform()方法 方式是onView(...).perform() 。也可以執(zhí)行多個(gè)操作在一個(gè)perform中如:perform(click(),clearText()) 。
所有的操作都有一個(gè)前提 ———— 就是要執(zhí)行的view必須在當(dāng)前界面上顯示出來(有且可見)。
| 方法名 | 含義 |
|---|---|
| click() | 點(diǎn)擊view |
| clearText() | 清除文本內(nèi)容 |
| swipeLeft() | 從右往左滑 |
| swipeRight() | 從左往右滑 |
| swipeDown() | 從上往下滑 |
| swipeUp() | 從下往上滑 |
| click() | 點(diǎn)擊view |
| closeSoftKeyboard() | 關(guān)閉軟鍵盤 |
| pressBack() | 按下物理返回鍵 |
| doubleClick() | 雙擊 |
| longClick() | 長按 |
| scrollTo() | 滾動(dòng) |
| replaceText() | 替換文本 |
| openLinkWithText() | 打開指定超鏈 |
ViewAssertions 檢驗(yàn)結(jié)果
使用check()方法來檢查View是否符合我們的期望: onView(...).check() 檢查view中是否含有文本“hello” check(matches(withText("hello")))
看下我寫的示例
我們基本所有的app都有登錄功能,都需要呼入用戶名和密碼,那么在點(diǎn)擊登錄之前需要對(duì)用戶名和密碼進(jìn)行非空、格式等驗(yàn)證。
以下示例我們點(diǎn)擊登錄按鈕時(shí),首先對(duì)輸入的用戶名和密碼進(jìn)行驗(yàn)證,驗(yàn)證不通過在TextView上顯示對(duì)應(yīng)原因,驗(yàn)證沒有問題顯示“登錄成功”。
Activity界面及邏輯
@Override
public void onClick(View view) {
if (view.getId() == R.id.bt_login) {
login();
}
}
/**
* 去登錄
*/
private void login() {
String name = et_name.getText().toString().trim();
String pwd = et_pwd.getText().toString().trim();
if (TextUtils.isEmpty(name)) {
tv_login_result.setText("用戶名為空");
return;
}
if (name.length() < 6 ) {
tv_login_result.setText("用戶名格式錯(cuò)誤");
return;
}
if (TextUtils.isEmpty(pwd)) {
tv_login_result.setText("密碼為空");
return;
}
if (pwd.length() < 6 ) {
tv_login_result.setText("密碼格式錯(cuò)誤");
return;
}
tv_login_result.setText("登錄成功");
}
其他代碼忽略。
@RunWith(AndroidJUnit4.class)
@LargeTest
public class MainActivityTest {
private String[] names = {"", "a", "123123"};
private String[] pwds = {"", "a", "123123"};
@Rule
public ActivityTestRule mTestRule = new ActivityTestRule<>(MainActivity.class);
@Before
public void init() {
Log.e("TAG", "init: ");
}
@Test
public void testLogin() {
// 不做任何輸入,直接點(diǎn)擊登錄
onView(allOf(withId(R.id.bt_login), isDisplayed())).perform(click());
onView(allOf(withId(R.id.tv_login_result), isDisplayed())).check(matches(withText("用戶名為空")));
// 用戶名是空,點(diǎn)擊登錄
onView(allOf(withId(R.id.et_name), isDisplayed())).perform(replaceText(names[0]), closeSoftKeyboard());
onView(allOf(withId(R.id.bt_login), isDisplayed())).perform(click());
onView(allOf(withId(R.id.tv_login_result), isDisplayed())).check(matches(withText("用戶名為空")));
// 用戶名格式錯(cuò)誤,點(diǎn)擊登錄
onView(allOf(withId(R.id.et_name), isDisplayed())).perform(replaceText(names[1]), closeSoftKeyboard());
onView(allOf(withId(R.id.bt_login), isDisplayed())).perform(click());
onView(allOf(withId(R.id.tv_login_result), isDisplayed())).check(matches(withText("用戶名格式錯(cuò)誤")));
// 用戶名和密碼都正確,點(diǎn)擊登錄
onView(allOf(withId(R.id.et_name), isDisplayed())).perform(replaceText(names[2]), closeSoftKeyboard());
onView(allOf(withId(R.id.et_pwd), isDisplayed())).perform(replaceText(pwds[2]), closeSoftKeyboard());
onView(allOf(withId(R.id.bt_login), isDisplayed())).perform(click());
onView(allOf(withId(R.id.tv_login_result), isDisplayed())).check(matches(withText("登錄成功")));
}
}
這里我們事先定義了一些測試數(shù)據(jù),使用Espresso進(jìn)行模擬各種情況輸入和點(diǎn)擊,測試是否符合我們的預(yù)期:

對(duì)Espresso的介紹大概就是這些了,希望大家多提建議,一起進(jìn)步。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
android計(jì)時(shí)器,時(shí)間計(jì)算器的實(shí)現(xiàn)方法
android計(jì)時(shí)器,時(shí)間計(jì)算器的實(shí)現(xiàn)方法,需要的朋友可以參考一下2013-03-03
雙緩沖技術(shù)實(shí)現(xiàn)Android 畫板應(yīng)用
這篇文章主要介紹了Android 采用雙緩存技術(shù)實(shí)現(xiàn)畫板應(yīng)用的相關(guān)資料,并附有代碼實(shí)例,有需要的小伙伴可以參考下2016-07-07
Android studio button 按鈕 四種綁定事件的方法【實(shí)例代碼】
這篇文章主要介紹了Android studio button 按鈕 四種綁定事件的方法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-08-08
OKhttp攔截器實(shí)現(xiàn)實(shí)踐環(huán)節(jié)源碼解析
這篇文章主要為大家介紹了OKhttp攔截器實(shí)現(xiàn)實(shí)踐環(huán)節(jié)源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
學(xué)習(xí)Android開發(fā)之RecyclerView使用初探
Android開發(fā)學(xué)習(xí)之路的第一課RecyclerView使用初探,感興趣的小伙伴們可以參考一下2016-07-07
Android調(diào)用相機(jī)并將照片存儲(chǔ)到sd卡上實(shí)現(xiàn)方法
Android中實(shí)現(xiàn)拍照有兩種方法,一種是調(diào)用系統(tǒng)自帶的相機(jī),還有一種是自己用Camera類和其他相關(guān)類實(shí)現(xiàn)相機(jī)功能,這種方法定制度比較高,需要的朋友可以了解下2012-12-12
Android自定義View之漸變色折線圖的實(shí)現(xiàn)
折線圖的實(shí)現(xiàn)方法在github上有很多開源的程序,但是對(duì)于初學(xué)者來講,簡單一點(diǎn)的教程可能更容易入門,下面這篇文章主要給大家介紹了關(guān)于Android自定義View之漸變色折線圖的相關(guān)資料,需要的朋友可以參考下2022-04-04
Android手機(jī)號(hào)碼輸入框(滿11位自動(dòng)跳到下個(gè)輸入框)實(shí)例代碼
這篇文章主要介紹了Android手機(jī)號(hào)碼輸入框(滿11位自動(dòng)跳到下個(gè)輸入框)實(shí)例代碼,需要的朋友可以參考下2017-10-10

