Android单元测试之对Activity的测试示例

发布时间 - 2026-01-11 02:48:44    点击率:

上一篇文章已经介绍了单元测试的作用和简单示例,如果不了解的读者可以先阅读上一篇Android单元测试-作用以及简单示例。 

这篇文章主要介绍常见的Activity中的测试。

对Acitivity的测试

对于Activity,我们大致有两种测试需求:

1、在Activity正常启动后,查看界面布局是否正确,包括View的点击事件等是否正确。

2、需要在Activity启动前完成各种数据的部署,然后查看Activity的效果。

对于这两种需求,笔者分别做了两个示例解说:

1、检测一个布局中的button和TextView是否正确。

2、从网络动态获取String到Activity界面显示,并且这个图片的URL是由Intent传递过来的。

环境部署

首先要导入expresso-core的包,如下:

dependencies {
  // Other dependencies ...
  androidTestCompile 'com.android.support.test.espresso:espresso-core:2.2.2'
}

当然在目前的项目架构中一般已经自动导入了这个包,所以不需要自己导入,笔者项目中自动导入的包如下如下:

dependencies {
  compile fileTree(include: ['*.jar'], dir: 'libs')
  androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
    exclude group: 'com.android.support', module: 'support-annotations'
  })
  compile 'com.android.support:appcompat-v7:26.0.0-alpha1'
  compile 'com.android.support.constraint:constraint-layout:1.0.2'
  testCompile 'junit:junit:4.12'

}

项目结构如下:

布局View的测试:

package com.example.xujiajia_sx.myexpressotest;

import android.app.Activity;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

/**
 * Created by xujiajia_sx on 2017/8/14.
 */

public class SimpleViewActivity extends Activity{
  private TextView tv;
  private Button btn;
  @Override
  protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.act_simple_view);

    initView();
  }

  private void initView() {
    tv=findViewById(R.id.tv_simple_view);
    btn=findViewById(R.id.btn_simple_view);

    tv.setText("111");
    btn.setText("222");

    btn.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View view) {
        tv.setText("777");
      }
    });
  }
}

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical">

  <TextView
    android:id="@+id/tv_simple_view"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
     />

  <Button
    android:id="@+id/btn_simple_view"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    />

</LinearLayout>
package com.example.xujiajia_sx.myexpressotest;

import android.support.test.rule.ActivityTestRule;

import org.junit.Rule;
import org.junit.Test;

import static android.support.test.espresso.Espresso.onView;
import static android.support.test.espresso.action.ViewActions.click;
import static android.support.test.espresso.assertion.ViewAssertions.matches;
import static android.support.test.espresso.matcher.ViewMatchers.withId;
import static android.support.test.espresso.matcher.ViewMatchers.withText;

/**
 * Created by xujiajia_sx on 2017/8/14.
 */

public class SimpleViewTest {

  @Rule
  public ActivityTestRule<SimpleViewActivity> mActivityTestRule =
      new ActivityTestRule<SimpleViewActivity>(SimpleViewActivity.class);

  @Test
  public void textViewTest() throws Exception {
    onView(withId(R.id.tv_simple_view))
        .check(matches(withText("111")));
  }

  @Test
  public void buttonTest() throws Exception {
    onView(withId(R.id.btn_simple_view))
        .check(matches(withText("222")))
        .perform(click());
    onView(withId(R.id.tv_simple_view))
        .check(matches(withText("777")));
  }
}

测试主要逻辑:

1、首先要使用ActivityTestRule初始化你要测试的Activity。

2、编写测试方法,测试View是否是我们预期的样子。

两个测试方法逻辑如下:

textViewTest():

在Activity中查找id为tv_simple_view的View,检查它的text是否为“111”。

buttonTest():

在Activity中查找id为btn_simple_view的View,检查它的text是否为“222”。然后执行点击事件,点击事件的逻辑是在Activity的OnCreate中设置的,是把TextView的text设置为777。在执行完点击事件后,测试方法中继续测试TextView的text是否为“777”。

读者可能阅读到对View的测试非常陌生,不用担心,此处主要要理解测试的逻辑即可,笔者会在下篇文章具体讲解View的各种测试方法。

网络获取String的Activity测试:

package com.example.xujiajia_sx.myexpressotest;

import android.app.Activity;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.widget.TextView;

/**
 * Created by xujiajia_sx on 2017/8/14.
 */

public class ActTestActivity extends Activity{
  private TextView tv;
  @Override
  protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.act_act_test);

    initView();
  }

  private void initView() {
    tv= findViewById(R.id.tv_act_test);
    new Thread(new Runnable() {
      @Override
      public void run() {
        String url =getIntent().getStringExtra("url");
        final String s=mHttpClient.getInstance().get(url);
        runOnUiThread(new Runnable() {
          @Override
          public void run() {
            tv.setText(s);
          }
        });
      }
    }).start();
  }
}

package com.example.xujiajia_sx.myexpressotest;

/**
 * Created by xujiajia_sx on 2017/8/14.
 */

public class mHttpClient {
  private static HttpURLConnectionClient mClient = null;

  public static void setClient(HttpURLConnectionClient client) {
    mClient = client;
  }

  public static HttpURLConnectionClient getInstance() {
    return mClient;
  }
}

package com.example.xujiajia_sx.myexpressotest;

import android.util.Log;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;

import static android.content.ContentValues.TAG;

/**
 * Created by xujiajia_sx on 2017/8/14.
 */

public class HttpURLConnectionClient {

  public String get(String url) {
    HttpURLConnection conn = null;
    try {
      URL mURL = new URL(url);
      conn = (HttpURLConnection) mURL.openConnection();
      conn.setRequestMethod("GET");
      conn.setConnectTimeout(2000);
      conn.connect();
      InputStream is = conn.getInputStream();
      StringBuilder sb = new StringBuilder();
      BufferedReader reader = new BufferedReader(new InputStreamReader(is));
      String line;
      while ((line = reader.readLine()) != null) {
        sb.append(line).append("\n");
      }
      reader.close();
      return sb.toString();
    } catch (IOException e) {
      Log.e(TAG, "network error for mini program ", e);
      return "";
    } finally {
      //最后将conn断开连接
      if (conn != null) {
        conn.disconnect();
      }
    }
  }


}

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical">

  <TextView
    android:id="@+id/tv_act_test"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    />


</LinearLayout>



package com.example.xujiajia_sx.myexpressotest;

import android.content.Intent;
import android.support.test.rule.ActivityTestRule;

import org.junit.Rule;
import org.junit.Test;

/**
 * Created by xujiajia_sx on 2017/8/14.
 */

public class ActTest {
  @Rule
  public ActivityTestRule<ActTestActivity> mActivityTestRule=
      new ActivityTestRule<ActTestActivity>(ActTestActivity.class){
        @Override
        protected Intent getActivityIntent() {
          Intent intent=new Intent();
          intent.putExtra("url","http://www.weather.com.cn/adat/sk/101310201.html");
          return intent;
        }

        @Override
        protected void beforeActivityLaunched() {
          mHttpClient.setClient(new HttpURLConnectionClient());
        }
      };

  @Test
  public void mTest() throws Exception{

    Thread.sleep(5000);
  }


}

网络获取不要忘记在AndroidManifest中加网络权限喔。

这个Activity的主要逻辑就是接收Intent,然后获取到传过来的url,接着通过网络获取到url的String,显示到TextView上。

主要测试逻辑:

首先还是要定义ActivityTestRule,确定使用哪个Activity。

与前一个例子不同的是,这里要重写ActivityTestRule的两个方法,getActivityIntent() 和beforeActivityLaunched()。顾名思义,一个是设置Activity获取到的Intent,另一个是设置Activity启动跟之前的准备工作。

笔者此处在getActivityIntent() 中设置了传递的url,在beforeActivityLaunched()设置的网络获取的方式。

有些读者可能会好奇为什么网络获取的方式不默认呢,而要通过setClient()来设置?

因为这样可以更方便我们测试,在正式的项目中,我们可能会需要在代码中加入log等操作,但是正式的代码一般我们是不会去修改的,但是我们可以继承它,重写某些方法,然后把它放到测试需要的地方。

在这里我们就可以继承HttpURLConnectionClient 这个类,然后把继承的子类使用setClient()来作为网络获取的方式。

总结

Activity的使用方法大致如此了,如果有更多需求的读者可以查看一下官方ActivityTestRule的Reference。

链接如下:https://developer.android.google.cn/reference/android/support/test/rule/ActivityTestRule.html

第一种使用方法中设计到了对View的测试,由于篇幅较大,本篇文章未能详细讲述,笔者会在下篇文章做一定讲解。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# Android  # Activity测试  # 单元测试  # 详解appium+python 启动一个app步骤  # Python脚本在Appium库上对移动应用实现自动化测试  # 详解Android单元测试最佳实践  # android开机自启动APP及使用adb命令测试方法  # Android利用Espresso进行UI自动化测试的方法详解  # 在Android打包中区分测试和正式环境浅析  # 浅谈Android单元测试的作用以及简单示例  # Android和iOS 测试五个最好的开源自动化工具  # Android 中构建快速可靠的 UI 测试  # 简单谈谈android studio 的单元测试  # Android Monkey压力测试详细介绍  # Ubuntu中为Android系统上实现内置C可执行程序测试Linux内核驱动程序  # Android App开发的自动化测试框架UI Automator使用教程  # Android自动测试工具Monkey的实现方法  # Android测试中Appium的一些错误解决技巧  # 是否正确  # 会在  # 重写  # 的是  # 是在  # 在这里  # 你要  # 子类  # 不需要  # 是由  # 把它  # 我们可以  # 上一  # 顾名思义  # 不了解  # 这篇文章  # 后将  # 上一篇  # 有两种 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何快速搭建高效可靠的建站解决方案?  Laravel如何保护应用免受CSRF攻击?(原理和示例)  家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?  网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?  Laravel如何处理和验证JSON类型的数据库字段  如何基于云服务器快速搭建网站及云盘系统?  Windows10如何更改计算机工作组_Win10系统属性修改Workgroup  浅谈javascript alert和confirm的美化  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  悟空识字如何进行跟读录音_悟空识字开启麦克风权限与录音  非常酷的网站设计制作软件,酷培ai教育官方网站?  南京网站制作费用,南京远驱官方网站?  北京网站制作公司哪家好一点,北京租房网站有哪些?  javascript中的try catch异常捕获机制用法分析  怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?  Python数据仓库与ETL构建实战_Airflow调度流程详解  夸克浏览器网页跳转延迟怎么办 夸克浏览器跳转优化  如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?  Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言  儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?  进行网站优化必须要坚持的四大原则  奇安信“盘古石”团队突破 iOS 26.1 提权  C++时间戳转换成日期时间的步骤和示例代码  Laravel如何实现API版本控制_Laravel API版本化路由设计策略  如何用美橙互联一键搭建多站合一网站?  海南网站制作公司有哪些,海口网是哪家的?  Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)  如何制作一个表白网站视频,关于勇敢表白的小标题?  Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层  惠州网站建设制作推广,惠州市华视达文化传媒有限公司怎么样?  最好的网站制作公司,网购哪个网站口碑最好,推荐几个?谢谢?  三星网站视频制作教程下载,三星w23网页如何全屏?  如何快速使用云服务器搭建个人网站?  Laravel如何配置Horizon来管理队列?(安装和使用)  Laravel怎么实现模型属性的自动加密  Laravel怎么清理缓存_Laravel optimize clear命令详解  手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?  Win11任务栏卡死怎么办 Windows11任务栏无反应解决方法【教程】  详解jQuery中基本的动画方法  如何快速建站并高效导出源代码?  齐河建站公司:营销型网站建设与SEO优化双核驱动策略  Laravel怎么写单元测试_PHPUnit在Laravel项目中的基础测试入门  Laravel如何将应用部署到生产服务器_Laravel生产环境部署流程  历史网站制作软件,华为如何找回被删除的网站?  宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法  JS弹性运动实现方法分析  JavaScript Ajax实现异步通信  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  如何用5美元大硬盘VPS安全高效搭建个人网站?  php 三元运算符实例详细介绍