1、运用JUnit4 进行单元测试

首先在工程的 src 文件夹内创建 test 和 test/java 文件夹。

打开工程的 build.gradle(Module:app)文件,添加JUnit4依赖,点击Gradle sync按钮。

build.gradle

 dependencies {
testCompile 'junit:junit:4.12'
}

(1)新建被测类:

 public class Calculator {

     public double sum(double a, double b){
// 假设先返回结果0
return 0;
}
}

(2)新建测试类:


import org.junit.Before;
import org.junit.Test;
import static org.junit.Assert.*;
public class CalculatorTest {

    private Calculator mCalculator;

    @Before
public void setUp() throws Exception {
mCalculator = new Calculator();
} @Test
public void testSum() throws Exception {
//断言:1+1 = 2
assertEquals(mCalculator.sum(1d, 1d), 2d);
}
}

这时候 右键 - testSum()方法,选择选择Run > testRun , 也可以通过命令行运行测试,在工程目录内输入:

 ./gradlew test

这时测试因为我写被测类的时候,返回的是0,所以跟期望的值不一样,就会失败。

这时,我们修改下Calculator.java的函数:

 public double sum(double a, double b){
return a + b;
}

保存,这时候再运行测试,成功,跟期望值一样。

总结:位于src/tests目录下的测试是运行在本地电脑Java虚拟机上的单元测试。

编写测试,实现功能使测试通过,然后再添加更多的测试,这种工作方式使快速迭代成为可能,我们称之为测试驱动开发

2、使用Mockito等mocking框架来mock测试。

Mock:

mock对象就是在调试期间用来作为真实对象的替代品。

mock测试就是在测试过程中,对那些不容易构建的对象用一个虚拟对象来代替测试的方法就叫mock测试。

打开工程的 build.gradle(Module:app)文件,添加Mockito依赖,点击Gradle sync按钮。

build.gradle

 dependencies {
// 单元测试
testCompile 'org.mockito:mockito-all:2.0.2-beta'
testCompile 'junit:junit:4.12'
}

关于mokito的使用,官网已经给出很详细了。我这里主要想记录下主要用到的。对于mvp架构下,测试presenter返回到view的数据,是否正确。

配合mvp模式下,利用本地json,模仿网络请求,进行模拟数据,查看presenter 处理逻辑是否正确。

public class PresenterTest {

    @Mock
Repository repository; // 网络请求
@Mock
View view; // view PresenterImpl presenter; @Before
public void setUp() throws Exception {
// 如果有使用到rxjava,可以在这里处理rxjava变成同步执行
MockitoAnnotations.initMocks(this); presenter = new PresenterImpl(view, repository);
} @Test
public void getOrderList() throws Exception{
String json = readAssetsJSON("get_list.json").optString("data");
List<ViewModel> list = JSON.parseArray(json, ViewModel.class); when(repository.getList(1)).thenReturn(Observable.just(list));
presenter.getList(1); ArgumentCaptor<ArrayList> captor = ArgumentCaptor.forClass(ArrayList.class);
verify(presenter.getView()).onGetDataByFinish(captor.capture()); List<ViewModel> viewModels = captor.getValue(); assertEquals(list.size(), viewModels.size());
}
}

get_list.json 是放到 src/test下的assets 目录下,这个目录在:打开工程的 build.gradle(Module:app)文件,android节点下的sourceSets节点下,配置下:

 sourceSets {
main {
.....
} test {
java.srcDirs = [ 'src/test/java']
}
}

其他的用法:参考下官方就行,因为官网是英文的,看不懂的,还可以前往:Mockito 中文文档

3、使用Robolectric 

Robolectric 是一个针对于Android SDK 的单元测试框架,使用它可以测试驱动你的Android应用程序的开发。测试用例只需要在JVM基础上就能运行起来。

打开工程的 build.gradle(Module:app)文件,添加Robolectric依赖,点击Gradle sync按钮。

build.gradle

 dependencies {
// 单元测试
testCompile 'org.robolectric:robolectric:3.1-rc1'
testCompile 'org.mockito:mockito-all:2.0.2-beta'
testCompile 'junit:junit:4.12'
}

1、创建一个WelcomeActivity, 点击登录,跳转到登录页面。

布局文件:

<?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">  
    <Button 
        android:id="@+id/btn_login"
android:text="click the btn" 
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>  
</LinearLayout> 

Activity:

public class WelcomeActivity extends Activity {  
    @Override 
    protected void onCreate(Bundle savedInstanceState) {       
   super.onCreate(savedInstanceState); 
        setContentView(R.layout.welcome_activity);  
        final View button = findViewById(R.id.btn_login); 
        button.setOnClickListener(new View.OnClickListener() {  
            @Override 
            public void onClick(View view) { 
                startActivity(new Intent(getContext(), LoginActivity.class));     
      }      
    });
     }
 }

测试的是:当用户点击登陆按钮后,我们启动了正常的intent。

由于Robolectric只是一个模拟的单元测试框架,LoginActivity并不会真正的启动,但是我们可以检查是否准确的发出了WelcomActivity的intent。

 @RunWith(RobolectricTestRunner.class) 
public class WelcomeActivityTest {  
    @Test 
    public void clickingLogin_shouldStartLoginActivity() { 
        WelcomeActivity activity = Robolectric.setupActivity(WelcomeActivity.class);       
      activity.findViewById(R.id.btn_login).performClick();  
        Intent expectedIntent = new Intent(activity, WelcomeActivity.class);         
     assertThat(shadowOf(activity).getNextStartedActivity()).isEqualTo(expectedIntent);  
   } 

判断点击后textview 文本变化是否正常等:

 @RunWith(RobolectricTestRunner.class)
public class WelcomeActivityTest { @Test
public void clickLoginButton() throws Exception {
Activity activity = Robolectric.buildActivity(WelcomeActivity.class).create().get(); Button btn_login = (Button) activity.findViewById(R.id.btn_login);
TextView tv_result = (TextView) activity.findViewById(R.id.tv_result); btn_login.performClick();
String resultsText = tv_result.getText().toString();
assertThat(resultsText, equalTo("Click the login button"));
}
}

对于控制activity的生命周期,Robolectric 2.2版本以后增加了控制activity方法:

 Activity activity = Robolectric.buildActivity(WelcomeActivity.class).create().get();

这会创建一个WelcomeActivity,并且已经调用了onCreate()。

(1)检查一些在onCreate()和onResume()之间发生的事件:

 ActivityController controller = 
          Robolectric.buildActivity(WelcomeActivity.class).create().start(); 
Activity activity = controller.get(); 
// assert that something hasn't happened 
activityController.resume(); 

(2)完整的生命周期:

 Activity activity = 
      Robolectric.buildActivity(WelcomeActivity.class).create().start().resume().visible().get(); 

(3)使用intent 来启动activity:

 Intent intent = new Intent(Intent.ACTION_VIEW); 
Activity activity = 
    Robolectric.buildActivity(WelcomeActivity.class).withIntent(intent).create().get();

(4)保存状态:

1 Bundle savedInstanceState = new Bundle(); 
2 savedInstanceState.putExtra("user_age", "18");
3 savedInstanceState.putExtra("user_name", "jay");
4 Activity activity = Robolectric.buildActivity(WelcomeActivity.class).create() 
5        .restoreInstanceState(savedInstanceState).get(); 

Android 单元测试(junit、mockito、robolectric)的更多相关文章

  1. Android:单元测试Junit的配置

    在实际开发中,开发android软件的过程需要不断地进行测试.而使用Junit测试框架,侧是正规Android开发的必用技术,在Junit中可以得到组件,可以模拟发送事件和检测程序处理的正确性.... ...

  2. Android单元测试Junit (一)

    1.在eclips中建立一个Android工程,具体信息如下: 2.配置单元测试环境,打开AndroidManifest.xml,具体代码如下所示: <?xml version="1. ...

  3. Android 单元测试Junit

  4. Android单元测试

    安卓单元测试总结文章,目测主要会cover以下的主题: 什么是单元测试 为什么要做单元测试 JUnit Mockito Robolectric Dagger2 一个具体的app例子实践 神秘的bonu ...

  5. Android单元测试: 首先,从是什么开始

    Android单元测试: 首先,从是什么开始 http://chriszou.com/2016/04/13/android-unit-testing-start-from-what.html 这是一系 ...

  6. Android studio下gradle Robolectric单元测试配置

    android studio下gradle Robolectric单元测试配置 1.Robolectric Robolectric是一个基于junit之上的单元测试框架.它并不依赖于Android提供 ...

  7. JUnit + Mockito 单元测试(二)

    摘自: http://blog.csdn.net/zhangxin09/article/details/42422643 版权声明:本文为博主原创文章,未经博主允许不得转载. 目录(?)[-] 入门 ...

  8. JUnit + Mockito 单元测试(二)(good)

    import org.junit.Test; import org.mockito.Matchers; import org.mockito.Mockito; import java.util.Lis ...

  9. 阿里知识储备之二——junit学习以及android单元测试

    一,junit框架 http://blog.csdn.net/afeilxc/article/details/6218908 详细见这篇博客 juit目前已经可以和maven项目进行集成和测试,而且貌 ...

随机推荐

  1. 关于JQ toggle 的注意事项

    1.9.1以后的版本,好像不支持 jq 的 toggle function的用法啦.

  2. &&&&数组去重方法总结&&&&&

    [数组去重]本文一共总结了5种方法: //方法一:sort方法 var ary = [1, 4, 2, 3, 1, 2, 2, 3, 3, 2, 5, 2, 1, 2];Array.prototype ...

  3. mysql mac  安装修改初始密码

    step1:苹果->系统偏好设置->最下边点mysql 在弹出页面中 关闭mysql服务(点击stop mysql server) step2:进入终端输入:cd /usr/local/m ...

  4. android 移动网络实时抓包

    2G.3G环境,那就必须root进去tcpdump 方式抓. 准备: 一.root CF-auto-root: http://autoroot.chainfire.eu/ 需要清理全部数据,注意备份 ...

  5. WebService -- Java 实现之 CXF ( 添加系统预定义的拦截器)

    1. 概述 CXF允许我们在webservice的in/out位置添加拦截器.拦截器有两大分类,一类是系统预定义的:另一类是自定义拦截器. 2. 在server端添加拦截器. JaxWsServerF ...

  6. 破解激活Win10无风险?激活后删除激活工具无影响===http://www.pconline.com.cn/win10/693/6932077_all.html#content_page_4

    1Windows激活:测试环境搭建 随着Windows 10的发布,许多用户都用上了这个新一代的操作系统.Windows 10有个最好的设置就是,只要你在已经激活的旧系统中升进行升级操作,就能获得一个 ...

  7. css文件 引用后不起作用

    你如果填写的是相对路径,那么检查一下路径是否正确. 如果相对路径正确,那么有可能你的css样式的层级错误(概率也不低),比如说图片的引用路径发生了改变等等. 要看你预览的浏览器是什么,我经常遇到IE预 ...

  8. STM32F412应用开发笔记之一:初识NUCLEO-F412ZG

    今天终于收到了期待已久的NUCLEO-F412ZG,感谢电子发烧友论坛! 近几年来基本都是在STM32平台上做一些设计开发工作.STM32F103.STM32F107.STM32F429等都应用过,但 ...

  9. lua相关笔记

    --[[ xpcall( 调用函数, 错误捕获函数 ); lua提供了xpcall来捕获异常 xpcall接受两个参数:调用函数.错误处理函数. 当错误发生时,Lua会在栈释放以前调用错误处理函数,因 ...

  10. iOS之开发小技巧

    1.xcode如何添加快捷代码 xcode添加快捷代码 属性 2.cocoapods安装 cocoapods安装 3.iOS真机调试 真机调试 4.命令行自动打包 xcrun -sdk iphoneo ...