Android编程单元测试实例详解(附源码)

   2015-11-25 0
核心提示:这篇文章主要介绍了Android编程单元测试,结合完整实例形式详细分析了Android单元测试的具体步骤与相关技巧,并附带完整实例代码供读者下载参考,需要的朋友可以参考下

本文实例讲述了Android编程单元测试。分享给大家供大家参考,具体如下:

完整实例代码代码点击此处本站下载

本文是在上一篇文章《java编程之单元测试(Junit)实例分析》的基础上继续讲解android的单元测试,android源码中引入了java单元测试的框架(android源码目录:libcore\junit\src\main\java\junit\framework中可见),然后在java单元测试框架的基础上扩展属于android自己的测试框架。android具体框架类的关系图如下:

Android编程单元测试实例详解(附源码)

从上图的类关系图中可以知道,通过android测试类可以实现对android中相关重要的组件进行测试(如Activity,Service,ContentProvider,甚至是application)。

其实在android源码中,基本上每个系统应用都自带一个测试工程,如下图的源码中settings(设置)模块:

Android编程单元测试实例详解(附源码)

上图的tests文件夹中就是settings模块自带的单元测试工程,有兴趣的读者可自行去研读一下源代码。

eclipse下(当然,前提是要保证eclipse中相关的android环境已经搭建好)进行android单元测试:

1.Application的测试:

新建一个android项目,在该android项目添加一个继承Application的类,代码如下:

package com.phicomm.hu; 
import android.app.Application; 
public class FxAndroidApplication extends Application 
{ 
 @Override 
 public void onCreate() 
 { 
  // TODO Auto-generated method stub 
  super.onCreate(); 
 } 
 @Override 
 public void onTerminate() 
 { 
  // TODO Auto-generated method stub 
  super.onTerminate(); 
 } 
 public String getFavourite() 
 { 
  return "I Love Java"; 
 } 
} 

Appication类创建好后,接着创建对应的测试工程:选中其所在的android工程---->鼠标右键----->new---->Android Test Project----->输入测试工程名--->next----->选择被测试的目标android工程(此处为FxAndroidApplication所在的android工程)。这样,一个测试工程就创建完成了。

通过eclipse创建自动生成的测试工程项目和android工程项目结构上没什么大的区别,主要是在AndroidManifest.xml中有变化,如下:

<xml version="1.0" encoding="utf-8"> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
 package="com.phicomm.hu.test" 
 android:versionCode="1" 
 android:versionName="1.0" > 
 <uses-sdk android:minSdkVersion="10" /> 
 <instrumentation 
  android:name="android.test.InstrumentationTestRunner" 
  android:targetPackage="com.phicomm.hu" /> 
 <application 
  android:icon="@drawable/ic_launcher" 
  android:label="@string/app_name" > 
  <uses-library android:name="android.test.runner" /> 
 </application> 
</manifest> 

在AndroidManifest.xml注册了相关的测试环境(这些是android独有的):<uses-library android:name="android.test.runner" />实现使用相关的运行测试类库,<instrumentation />中的targetPackage为被测试类所在的包。

接下来在测试工程中创建FxAndroidApplicationd的测试类,代码如下:

package com.phicomm.hu.test; 
import com.phicomm.hu.FxAndroidApplication; 
import android.app.Application; 
import android.test.ApplicationTestCase; 
public class FxApplicationTest extends ApplicationTestCase<FxAndroidApplication>
{ 
 private FxAndroidApplication AppTest; 
 public FxApplicationTest() 
 { 
  //调用父类构造函数,且构造函中传递的参数为被测试的类 
  super(FxAndroidApplication.class); 
 } 
 @Override 
 protected void setUp() throws Exception 
 { 
  // TODO Auto-generated method stub 
  super.setUp(); 
  //获取application之前必须调用的方法 
  createApplication(); 
  //获取待测试的FxAndroidApplication 
  AppTest = getApplication(); 
 } 
 //测试FxAndroidApplication的getFavourite方法 
 public void testGetFavourite() 
 { 
  /*验证预测值"I Love C++"是否等于实际值, 
  由于实际值为"I love Java",所以此处测试结果为Failure*/ 
  assertEquals("I Love C++", AppTest.getFavourite()); 
 } 
}

测试类创建好后,就可以实现对FxAndroidApplicationd进行测试了。

测试方法:

启动android模拟器(也可以通过android手机)----->运行android工程----->在测试工程中选中测试类FxApplicationTest---->鼠标右键--->Run As---->Android Junit Test。这样,测试结果就可以在eclipse的Junit视图上显示了,如下图:

Android编程单元测试实例详解(附源码)

通过上图的测试结果可知,ApplicationTestCase测试类中有两个测试方法是默认进行测试的(testGetFavourite才是我们要测试的方法)。

当然,还可以通过adb进行测试:连接android手机------>打开电脑命令窗口(开始-->运行--->输入cmd)---->在命令窗口输入adb shell---->am instrument -w com.phicomm.hu.test(测试用例所在的包名)/android.test.InstrumentationTestRunner。

2.Activity的测试:

和上面application一样,先创建一个android工程,该工程中创建了两个activity,一个activity实现输入用户信息的登录界面,另一个acticity显示输入的用户信息。

效果图如下:

Android编程单元测试实例详解(附源码)

登录界面FxLoginActivity的代码如下:

package com.phicomm.hu; 
import android.app.Activity; 
import android.content.Intent; 
import android.os.Bundle; 
import android.view.View; 
import android.view.View.OnClickListener; 
import android.widget.Button; 
import android.widget.EditText; 
public class FxLoginActivity extends Activity 
{ 
 private EditText userName; 
 private EditText passWord; 
 /** Called when the activity is first created. */ 
 @Override 
 public void onCreate(Bundle savedInstanceState) 
 { 
  super.onCreate(savedInstanceState); 
  setContentView(R.layout.main); 
  userName = (EditText)findViewById(R.id.name); 
  passWord = (EditText)findViewById(R.id.psd); 
  Button login = (Button)findViewById(R.id.login); 
  Button reset = (Button)findViewById(R.id.reset); 
   //监听登录按钮 
   login.setOnClickListener(new OnClickListener() { 
   @Override 
   public void onClick(View v) 
   { 
    // TODO Auto-generated method stub 
    Intent intent = new Intent(FxLoginActivity.this, FxResultActivity.class);     
    //通过intent传递登录信息到ResultActivity的界面中显示 
    intent.putExtra("userName", userName.getText().toString()); 
    intent.putExtra("passWord", passWord.getText().toString()); 
    //启动ResultActivity显示登录界面信息 
    startActivity(intent); 
   } 
  }); 
   //监听重置按钮   
   reset.setOnClickListener(new OnClickListener() 
   { 
   @Override 
   public void onClick(View v) 
   { 
    // TODO Auto-generated method stub 
    resetInput(); 
   } 
  }); 
 } 
 public void resetInput() 
 { 
  userName.setText(""); 
  passWord.setText(""); 
 } 
}

main.xml布局文件的代码如下:

<xml version="1.0" encoding="utf-8"> 
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
 android:layout_width="fill_parent" 
 android:layout_height="fill_parent" 
 android:orientation="vertical" > 
 <EditText 
  android:id="@+id/name" 
  android:layout_width="fill_parent" 
  android:layout_height="wrap_content" 
  android:hint="@string/name"/> 
  <EditText 
  android:id="@+id/psd" 
  android:layout_width="fill_parent" 
  android:layout_height="wrap_content" 
  android:hint="@string/psd"/> 
  <LinearLayout 
   android:orientation="horizontal" 
   android:layout_width="match_parent" 
   android:layout_height="wrap_content" 
   > 
   <Button 
    android:id="@+id/login" 
    android:layout_width="fill_parent" 
    android:layout_height="wrap_content" 
    android:layout_weight="1" 
    android:text="@string/login"/> 
   <Button 
    android:id="@+id/reset" 
    android:layout_width="fill_parent" 
    android:layout_height="wrap_content" 
    android:layout_weight="1" 
    android:text="@string/reset"/> 
  </LinearLayout> 
</LinearLayout>

显示用户信息界面的FxResultActivity代码如下:

package com.phicomm.hu; 
import android.app.Activity; 
import android.content.Intent; 
import android.os.Bundle; 
import android.util.Log; 
import android.widget.EditText; 
import android.widget.TextView; 
public class FxResultActivity extends Activity 
{ 
 private static final String TAG = "ResultActivity"; 
 @Override 
 protected void onCreate(Bundle savedInstanceState) 
 { 
  // TODO Auto-generated method stub 
  super.onCreate(savedInstanceState); 
  setContentView(R.layout.result); 
  TextView result = (TextView)findViewById(R.id.result); 
  //通过得到intent获取登录界面传来的信息 
  Intent intent = getIntent(); 
  String userName = intent.getStringExtra("userName"); 
  String passWord = intent.getStringExtra("passWord"); 
  //将登录信息在页面中显示 
  result.setText("用户名:" + userName + "\n" + "密码:" + passWord);  
 } 
} 

以上的android工程创建好后,创建一个对应的测试工程:

测试工程中对应的FxLoginActivity类的测试代码如下(详细的代码讲解见代码中的相关注释,这里不在累赘):

package com.phicomm.hu.test; 
import android.app.Instrumentation; 
import android.test.ActivityInstrumentationTestCase2; 
import android.view.KeyEvent; 
import android.widget.Button; 
import android.widget.EditText; 
import com.phicomm.hu.FxLoginActivity; 
public class FxLoginActivityTest extends ActivityInstrumentationTestCase2<FxLoginActivity> 
{ 
 private Instrumentation mInstrumentation; 
 private FxLoginActivity mLoginTest; 
 private EditText userName; 
 private EditText passWord; 
 private Button login; 
 private Button reset; 
 public FxLoginActivityTest() 
 { 
  super(FxLoginActivity.class); 
 } 
 //重写setUp方法,在该方法中进行相关的初始化操作 
 @Override 
 protected void setUp() throws Exception 
 { 
  // TODO Auto-generated method stub 
  super.setUp(); 
  /**这个程序中需要输入用户信息和密码,也就是说需要发送key事件, 
   * 所以,必须在调用getActivity之前,调用下面的方法来关闭 
   * touch模式,否则key事件会被忽略 
   */ 
  //关闭touch模式 
  setActivityInitialTouchMode(false); 
  mInstrumentation = getInstrumentation(); 
  //获取被测试的FxLoginActivity 
  mLoginTest = getActivity(); 
  //获取FxLoginActivity相关的UI组件 
  userName = (EditText)mLoginTest.findViewById(com.phicomm.hu.R.id.name); 
  passWord = (EditText)mLoginTest.findViewById(com.phicomm.hu.R.id.psd); 
  login = (Button)mLoginTest.findViewById(com.phicomm.hu.R.id.login); 
  reset = (Button)mLoginTest.findViewById(com.phicomm.hu.R.id.reset);  
 } 
 //该测试用例实现在测试其他用例之前,测试确保获取的组件不为空 
 public void testPreConditions() 
 { 
  assertNotNull(mLoginTest); 
  assertNotNull(userName); 
  assertNotNull(passWord); 
  assertNotNull(login); 
  assertNotNull(reset); 
 } 
 /**该方法实现在登录界面上输入相关的登录信息。由于UI组件的 
  * 相关处理(如此处的请求聚焦)需要在UI线程上实现, 
  * 所以需调用Activity的runOnUiThread方法实现。 
  */ 
 public void input() 
 { 
  mLoginTest.runOnUiThread(new Runnable() 
  { 
   @Override 
   public void run() 
   { 
    // TODO Auto-generated method stub 
    userName.requestFocus(); 
    userName.performClick(); 
   } 
  }); 
  /*由于测试用例在单独的线程上执行,所以此处需要同步application, 
   * 调用waitForIdleSync等待测试线程和UI线程同步,才能进行输入操作。 
   * waitForIdleSync和sendKeys不允许在UI线程里运行 
   */ 
  mInstrumentation.waitForIdleSync(); 
  //调用sendKeys方法,输入用户名 
  sendKeys(KeyEvent.KEYCODE_P, KeyEvent.KEYCODE_H, 
    KeyEvent.KEYCODE_I, KeyEvent.KEYCODE_C, 
    KeyEvent.KEYCODE_O, KeyEvent.KEYCODE_M, 
    KeyEvent.KEYCODE_M); 
  mLoginTest.runOnUiThread(new Runnable() 
  { 
   @Override 
   public void run() 
   { 
    // TODO Auto-generated method stub 
    passWord.requestFocus(); 
    passWord.performClick(); 
   } 
  }); 
  //调用sendKeys方法,输入密码 
  sendKeys(KeyEvent.KEYCODE_1, KeyEvent.KEYCODE_2, 
    KeyEvent.KEYCODE_3, KeyEvent.KEYCODE_4); 
 } 
 //测试输入的用户信息 
 public void testInput() 
 { 
  //调用测试类的input方法,实现输入用户信息(sendKeys实现输入) 
  input(); 
  //测试验证用户信息的预期值是否等于实际值 
  assertEquals("phicomm", userName.getText().toString()); 
  //密码的预期值123与实际值1234不符,Failure; 
  assertEquals("123", passWord.getText().toString()); 
 } 
 //测试登录按钮 
 public void testLogin() 
 { 
  input(); 
  //开新线程,并通过该线程在实现在UI线程上执行操作 
  mInstrumentation.runOnMainSync(new Runnable() 
  { 
   @Override 
   public void run() 
   { 
    // TODO Auto-generated method stub 
    login.requestFocus(); 
    login.performClick(); 
   } 
  }); 
 } 
 //测试重置按钮 
 public void testReset() 
 { 
  input(); 
  mInstrumentation.runOnMainSync(new Runnable() 
  { 
   @Override 
   public void run() 
   { 
    // TODO Auto-generated method stub 
    reset.requestFocus(); 
    //点击按钮 
    reset.performClick(); 
   } 
  }); 
  //验证重置按钮的实现功能,是否点击后内容为空 
  assertEquals("", userName.getText().toString()); 
  assertEquals("", passWord.getText().toString()); 
 } 
} 

运行该测试类进行测试(选中---->Run As--->Android Junit Test),然后会自动启动模拟器进行相关的输入点击测试。注:测试时可以发现,程序在测试到testLogin()方法登录到另一个界面时,测试就停止了,也就是说testReset()没测试到。所以,需要测试testReset()时可以先把testLogin()注释掉,不然程序会测试到testLogin()后就不在对testReset()进行测试。

FxResultActivity的测试类代码如下:

package com.phicomm.hu.test; 
import android.content.Intent; 
import android.test.ActivityInstrumentationTestCase2; 
import android.widget.TextView; 
import com.phicomm.hu.FxResultActivity; 
public class FxResultActivityTest extends ActivityInstrumentationTestCase2<FxResultActivity> 
{ 
 private static final String LOGIN_INFO = "用户名:feixun\n密码:123"; 
 private FxResultActivity mResultActivity; 
 private TextView result; 
 public FxResultActivityTest() 
 { 
  super(FxResultActivity.class); 
 } 
 @Override 
 protected void setUp() throws Exception 
 { 
  // TODO Auto-generated method stub 
  super.setUp(); 
  //创建Intent,通过Intent传递用户的登录信息 
  Intent intent = new Intent(); 
  intent.putExtra("userName", "feixun"); 
  intent.putExtra("passWord", "123"); 
  //通过携带用户登录信息的intent启动FxResultActivity 
  mResultActivity = launchActivityWithIntent("com.phicomm.hu", 
    FxResultActivity.class, intent); 
  //获取UI组件 
  result = (TextView)mResultActivity.findViewById(com.phicomm.hu.R.id.result); 
 } 
 //测试验证用户的登录信息 
 public void testLoginInfo() 
 { 
  //验证预期值是否等于实际值 
  assertEquals(LOGIN_INFO, result.getText().toString()); 
 } 
}

运行上面的测试类,结果正确。

希望本文所述对大家Android程序设计有所帮助。

 
标签: Android 单元测试
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • 说一说Android Studio和IDEA中一个很有用的内存调试插件
    说一说Android Studio和IDEA中一个很有用的内存
    JetBrains JVM Debugger Memory View plugin 在我最近的研发活动期间寻找新的工具,以提高我的开发经验,使Android Studio的生活更轻松,我发现一个有用的插件,我从来没有听说过。 这就是为什么,我决定写这个强大的工具,它如何帮助我与内存调试我的应用程
  • 安卓中通知功能的具体实现
    安卓中通知功能的具体实现
    通知[Notification]是Android中比较有特色的功能,当某个应用程序希望给用户发出一些提示信息,而该应用程序又不在前台运行时,就可以借助通知实现。使用通知的步骤1、需要一个NotificationManager来获得NotificationManager manager = (NotificationManager
    02-05 安卓开发
  • Android view系统分析-setContentView
    Android view系统分析-setContentView
    第一天上班,列了一下今年要学习的东西。主要就是深入学习Android相关的系统源代码,夯实基础。对于学习Android系统源代码,也没什么大概,就从我们平常使用最基础的东西学起,也就是从view这个切入点开始学习Android的源码,在没分析源码之前,我们有的时候
    02-05 安卓开发
  • 如何进行网络视频截图/获取视频的缩略图
    如何进行网络视频截图/获取视频的缩略图
    小编导读:获取视频的缩略图,截图正在播放的视频某一帧,是在音视频开发中,常遇到的问题。本文是主要用于点播中截图视频,同时还可以获取点播视频的缩略图进行显示,留下一个问题,如下图所示, 如果要获取直播中节目视频缩略图,该怎么做呢?(ps:直播是直
  • Android NDK 层发起 HTTP 请求的问题及解决
    Android NDK 层发起 HTTP 请求的问题及解决
    前言新的一年,大家新年快乐~~鸡年大吉!本次给大家带来何老师的最新文章~虽然何老师还在过节,但依然放心不下广大开发者,在此佳节还未结束之际,给大家带来最新的技术分享~ 事件的起因不说了,总之是需要实现一个 NDK 层的网络请求。为了多端适用,还是选择
  • SDK热更之如何在SDK代码中自动插桩及如何生成补
    写在前面本文是SDKHotfix相关的SDK热更系列文章中的一篇,以下为项目及系列文章相关链接:SDKHotfix整体介绍:http://blog.bihe0832.com/sdk_hotfix_project.htmlSDKHotfix对应github地址:https://github.com/bihe0832/SDKHoxFix这篇文章主要介绍一下SDK热更
  • 安装量破千万的第一个产品,我总结了3句话
    安装量破千万的第一个产品,我总结了3句话
    在今天的文章中,作者回顾了自己的第一个产品,他说“我做的第一款产品,是我的一块里程碑。”一起来看看~背景老牌大型互联网公司,部门内部创业的一个项目。我作为产品经理,也是第一次做产品经理,主导产品项目。实际上,项目初期包括我和安卓开发2个人。开
  • 移动周刊第 176 期:Android 知识梳理
    移动周刊第 176 期:Android 知识梳理
    写在前面 本期移动周刊第 176 期如约而至,聚焦 Android、iOS、VR/AR/MR、直播等前沿移动开发技术,收录一周最热点,解读开发技巧,每周三移动周刊抢先看,我们希望从中能够让你有一些收获,如果你有好的文章以及优化建议,请发送邮件至mobilehub@csdn.net,
  • Android插件化(六): OpenAtlasの改写aapt以防止资源ID冲突
    Android插件化(六): OpenAtlasの改写aapt以防
    引言Android应用程序的编译中,负责资源打包的是aapt,如果不对打包后的资源ID进行控制,就会导致插件中的资源ID冲突。所以,我们需要改写aapt的源码,以达到通过某种方式传递资源ID的Package ID,通过aapt打包时获取到这个Package ID并且应用才插件资源的命名
    02-05 安卓开发
  • Android架构(一)MVP架构在Android中的实践
    Android架构(一)MVP架构在Android中的实践
    为什么要重视程序的架构设计 对程序进行架构设计的原因,归根结底是为了 提高生产力 。通过设计是程序模块化,做到模块内部的 高聚合 和模块之间的 低耦合 (如依赖注入就是低耦合的集中体现)。 这样做的好处是使得程序开发过程中,开发人员主需要专注于一点,
    02-05 安卓开发
点击排行