好用的安卓自动化测试框架Espresso

 

2024软件测试面试刷题,这个小程序(永久刷题),靠它快速找到工作了!(刷题APP的天花板)-CSDN博客跳槽涨薪的朋友们有福了,今天给大家推荐一个软件测试面试的刷题小程序。https://blog.csdn.net/weixin_54696666/article/details/139783730?spm=1001.2014.3001.5501icon-default.png?t=N7T8https://blog.csdn.net/weixin_54696666/article/details/139783730?spm=1001.2014.3001.5501

Android测试概述

  Android自动化测试是应用开发中提升开发效率的一种方式,它可以在短时间内跑完所有的测试方案以及实现各种交互,从而让开发者更加方便和直观的知道代码的实现效果。

  Android自动化测试可以分为单元测试、UI测试、Monkey测试三种,同时也分小中大三种测试。

  单元测试

  单元测试又称为模块测试,是针对程序模块(软件设计的最小单位)来进行正确性检验的测试工作,程序单元是应用的最小可测试部件。

  Android中的单元测试是基于JUnit,可分为本地测试和Instrumented 测试。

  本地测试对应的项目目录是
module-name/src/test/java/,本地测试是运行在本地JVM上的,不需要模拟器或者设备的支持Instrumented 测试在项目中对应的目录是module-name/src/androidTest/java/,这个测试包含了Android系统的api,所以要在模拟器或者设备上运行,速度比本地测试慢。

  单元测试最常用的框架就是JUnit跟Mockito组合的测试框架,其中JUnit用来跑主要的测试代码,Mockito可以创建特定测试对象(非对象本身)并设置一些特定的属性供测试使用。

  UI测试

  UI测试是测试应用中的各种交互是否达到了实现的效果。常用的UI测试框架有Espresso和UIAutomator。

  UIAutomator是一个界面测试框架,适用于整个系统上以及多个已安装应用间的跨应用功能界面测试,它提供了一组 API,用于构建在用户应用和系统应用上执行交互的界面测试。

  Espresso是Google的开源自动化测试框架。相对于UIAutomator,它的特点是规模更小、更简洁,API更加精确,编写测试代码简单,容易快速上手。因为他是基于Instrumentation的,所以不能跨App。

  Espresso的使用

  Espresso有三个重要的类:分别是Matchers(匹配器)、ViewAction(界面行为)、ViewAssertions(界面判断)。

  其中Matchers是常常是通过匹配条件来找UI组件或过滤UI,而ViewAction是来模拟用户操作界面的行为,ViewAssertions对模拟行为操作的View进行变换和结果验证。

  执行顺序如下:

  依赖和配置

  在导入好包之后就可以module的AndroidTest包下开始编写自己的UI自动化测试代码了。

  测试界面跳转

  测试界面跳转时需要用的Espresso-Intents库,它是Espresso的扩展程序,支持对被测应用发出的intent进行验证和打桩。

  它与Mockit 类似,但适用于Android Intent,Espresso-Intents有两个比较重要的方法itending()和intended()。

  intending() 该方法可以插桩(设置跳到其他Activity后返回的参数),原来的Activity调用startActivityForResult()触发桩测试。

  intended() 方法是对发送出去的intent进行确认,如果不是给出的intent则会报错。

@Test fun validateIntentSentToPackage() {

user.clickOnView(system.getView(R.id.callButton))

intended(toPackage("com.android.phone"))

}

  异步测试

  Espresso测试有个很强大之处就是它在多个测试操作中是线程安全的,它会等待当前进程的消息队列中的UI事件,并且在任何一个测试操作中会等待其中的异步任务结束才会执行下一个测试。

  Espresso异步测试是要导入以下两个包:

  异步测试的流程

  新建一个类继承IdlingResource:

public class SimpleIdlingResource implements IdlingResource {

private final String mResourceName;

//这个counter值就像一个标记,默认为0

private final AtomicInteger counter = new AtomicInteger(0);

private volatile ResourceCallback resourceCallback;

public SimpleIdlingResource(String resourceName) {

mResourceName = resourceName;

}

@Override

public String getName() {

return mResourceName;

}

@Override

public boolean isIdleNow() {

return counter.get() == 0;

}

@Override

public void registerIdleTransitionCallback(ResourceCallback resourceCallback) {

this.resourceCallback = resourceCallback;

}

//每当我们开始异步请求,把counter值+1

public void increment() {

counter.getAndIncrement();

}

//当我们获取到网络数据后,counter值-1;

public void decrement() {

int counterVal = counter.decrementAndGet();

//如果这时counter == 0,说明异步结束,执行回调。

if (counterVal == 0) {

//

if (null != resourceCallback) {

resourceCallback.onTransitionToIdle();

}

}

if (counterVal < 0) {

//如果小于0,抛出异常

throw new IllegalArgumentException("Counter has been corrupted!");

}

}

}

  新建一个管理类去管理上述类:

public class EspressoIdlingResource {

private static final String RESOURCE = "GLOBAL";

private static SimpleIdlingResource mCountingIdlingResource =

new SimpleIdlingResource(RESOURCE);

public static void increment() {

mCountingIdlingResource.increment();

}

public static void decrement() {

mCountingIdlingResource.decrement();

}

public static IdlingResource getIdlingResource() {

return mCountingIdlingResource;

}

}

  将管理类插入到异步代码中,在异步任务开始前插入代码:

protected void onReqStart(int where) {

//异步开始前

EspressoIdlingResource.increment();

}

  异步任务结束后插入代码:

private void onFinish() {

if(!EspressoIdlingResource.getIdlingResource().isIdleNow()){

EspressoIdlingResource.decrement();

}

}

  在测试代码中注册IdlingRecource:

@Before

public void setUp() throws Exception {

//调用Activity中我们已经设置好的getIdlingresource()方法,获取Idlingresource对象

idlingresource = activityRule.getActivity().getIdlingresource();

//去掉下行注释,只有异步结束后,才进行接下来的测试代码(tests passed)

//注册异步监听,当该idlingresource中的counter标记值为0时才进行接下来的测试代码

Espresso.registerIdlingResources(idlingresource);

@Test

public void onLoadingFinished() throws Exception {

// 不再需要这样的代码

// Thread.sleep(5000);

// 未注册idlingResource时,立即进行test,此时异步并未结束,报错(tests failed)

onView(withId(R.id.text))

.check(matches(withText("success!")));

@After

public void release() throws Exception {

//我们在测试结束后取消注册,释放资源

Espresso.unregisterIdlingResources(idlingresource);

  测试过程中遇到的问题

  ViewAction()中的typeText()使用。

  如果直接调用typeText(),那么它的使用对象就一定是editText,如果是自定义的未继承editText的view使用typeText()时,要在使用前supportsInputMethods(),否则会报错误:

onView(allOf(supportsInputMethods(), isDescendantOfA(withId(R.id.test)))).perform(typeText("111111"))

  在调用完typeText()后最好调用一下 closeSoftKeyboard(),否则会出现某些view在点击时被阻挡而报错。

总结

  自动化测试的目的不仅仅是解放双手,它也可以让开发者提前知道某些小Bug,不用到了测试阶段才被提出来,这样就提升了开发的效率。

最后: 下方这份完整的软件测试视频教程已经整理上传完成,需要的朋友们可以自行领取【保证100%免费】

​​​软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

在这里插入图片描述

在这里插入图片描述

  • 23
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值