`
MyEyeOfJava
  • 浏览: 1159494 次
  • 性别: Icon_minigender_1
  • 来自: 北京
博客专栏
7af2d6ca-4fe1-3e9a-be85-3f65f7120bd0
测试开发
浏览量:71418
533896eb-dd7b-3cde-b4d3-cc1ce02c1c14
晨记
浏览量:0
社区版块
存档分类
最新评论

[Android]单元测试实例

阅读更多
转自:http://yuanzhifei89.iteye.com/blog/1122104  作者:yuanzhifei

测试相关资源
让开发自动化: 用 Eclipse 插件提高代码质量http://www.ibm.com/developerworks/cn/java/j-ap01117/index.html

代码测试覆盖率介绍:http://www.cnblogs.com/coderzh/archive/2009/03/29/1424344.html


学习android单元测试时遇到的一些问题:
1.开始以为单元测试一定要从程序的launch Activity,一步一步的跳转到所要测试的Activity能对其进行测试。
但实际上,我们可以从任意一个activity开始,对任意一个activity进行测试。

2.在运行单元测试之前,一定要先将要测试的程序安装到模拟器或真机上。

junit相关
android中的测试框架是扩展的junit3,所以在学习android中的单元测试签,可以先熟悉下junit3的使用,junit3要学习的东西应该并不多,就几页纸的东西。入门可以参考这个:http://android.blog.51cto.com/268543/49994

android单元测试框架中涉及的注解
@Suppress 可以用在类或这方法上,这样该类或者该方法就不会被执行
@UiThreadTest 可以用在方法上,这样该方法就会在程序的ui线程上执行
@LargeTest, @MediumTest, @SmallTest 用在方法上,标记所属的测试类型,主要是用于单独执行其中的某一类测试时使用。具体参考InstrumentationTestRunner类的文档。
@Smoke 具体用法还不清楚

android单元测试框架中涉及的一些类的uml




接下来我们以demo project来讲解如何使用android中的单元测试
主要包括了三个activity:
MainActivity:仅包含一个button,点击后就可以进入LoginActivity

LoginActivity:可以输入username, password,然后点击submit的话可进入HomeActivity,如果点击reset的话,输入的内容就会被清空

HomeActivity:在TextView中显示LoginActivity输入的内容

首先我们创建要测试的项目demo(使用了2.1)









MainActivity代码

public class MainActivity extends Activity {  
    private static final boolean DEBUG = true;  
    private static final String TAG = "-- MainActivity";  
  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        if (DEBUG) {  
            Log.i(TAG, "onCreate");  
        }  
  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.act_main);  
        View toLoginView = findViewById(R.id.to_login);  
        toLoginView.setOnClickListener(new View.OnClickListener() {  
            public void onClick(View view) {  
                if (DEBUG) {  
                    Log.i(TAG, "toLoginView clicked");  
                }  
  
                Intent intent = new Intent(getApplicationContext(), LoginActivity.class);  
                startActivity(intent);  
            }  
        });  
    }  
}  


MainActivity的布局文件

<LinearLayout  
    xmlns:android="http://schemas.android.com/apk/res/android"  
    android:layout_width="fill_parent"  
    android:layout_height="fill_parent"  
>  
    <Button  
        android:id="@+id/to_login"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:layout_gravity="bottom"  
        android:text="to login" />  
</LinearLayout>  




LoginActivity代码
public class LoginActivity extends Activity {  
    private static final boolean DEBUG = true;  
    private static final String TAG = "-- LoginActivity";  
  
    private EditText mUsernameView;  
    private EditText mPasswordView;  
  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        if (DEBUG) {  
            Log.i(TAG, "onCreate");  
        }  
  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.act_login);  
        mUsernameView = (EditText) findViewById(R.id.username);  
        mPasswordView = (EditText) findViewById(R.id.password);  
  
        View submitView = findViewById(R.id.submit);  
        submitView.setOnClickListener(new View.OnClickListener() {  
            public void onClick(View view) {  
                if (DEBUG) {  
                    Log.i(TAG, "submitView clicked");  
                }  
  
                Intent intent = new Intent(getApplicationContext(), HomeActivity.class);  
                intent.putExtra(HomeActivity.EXTRA_USERNAME, mUsernameView.getText().toString());  
                intent.putExtra(HomeActivity.EXTRA_PASSWORD, mPasswordView.getText().toString());  
                startActivity(intent);  
            }  
        });  
  
        View resetView = findViewById(R.id.reset);  
        resetView.setOnClickListener(new View.OnClickListener() {  
            public void onClick(View view) {  
                if (DEBUG) {  
                    Log.i(TAG, "resetView clicked");  
                }  
  
                mUsernameView.setText("");  
                mPasswordView.setText("");  
                mUsernameView.requestFocus();  
            }  
        });  
    }  
}  

LoginActivity的布局文件
<LinearLayout  
    xmlns:android="http://schemas.android.com/apk/res/android"  
    android:layout_width="fill_parent"  
    android:layout_height="fill_parent"  
    android:orientation="vertical"  
>  
    <TextView  
        android:id="@+id/label_username"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:text="username:" />  
          
    <EditText  
        android:id="@+id/username"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:inputType="text" />  
          
    <TextView  
        android:id="@+id/label_password"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:text="password:" />  
          
    <EditText  
        android:id="@+id/password"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:inputType="textPassword" />  
          
    <Button  
        android:id="@+id/submit"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:text="submit" />  
          
    <Button  
        android:id="@+id/reset"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:text="reset" />  
</LinearLayout>  




HomeActivity代码
public class HomeActivity extends Activity {  
    private static final boolean DEBUG = true;  
    private static final String TAG = "-- HomeActivity";  
  
    public static final String EXTRA_USERNAME = "yuan.activity.username";  
    public static final String EXTRA_PASSWORD = "yuan.activity.password";  
  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        if (DEBUG) {  
            Log.i(TAG, "onCreate");  
        }  
        super.onCreate(savedInstanceState);  
        Intent intent = getIntent();  
        StringBuilder sb = new StringBuilder();  
        sb.append("username:").append(intent.getStringExtra(EXTRA_USERNAME)).append("\n");  
        sb.append("password:").append(intent.getStringExtra(EXTRA_PASSWORD));  
  
        setContentView(R.layout.act_home);  
        TextView loginContentView = (TextView) findViewById(R.id.login_content);  
        loginContentView.setText(sb.toString());  
    }  
}  


HomeActivity的布局文件
<LinearLayout  
    xmlns:android="http://schemas.android.com/apk/res/android"  
    android:layout_width="fill_parent"  
    android:layout_height="fill_parent"  
>  
    <TextView  
        android:id="@+id/login_content"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:layout_gravity="center_vertical"  
        android:gravity="center"  
        android:textColor="#EEE"  
        android:textStyle="bold"  
        android:textSize="25sp" />  
</LinearLayout>  



程序非常简单,接下来我们为demo创建单元测试工程demo_unittest


选择之前创建的工程demo,然后eclipse会自动帮我们设定api level,包名等。(测试用例的包名一般就是在要测试类的包名后加上test)




创建完后eclipse会自动为我们创建好所需的目录,Manifest.xml文件


接下来就是为要测试的类编写测试用例。关于要用哪个测试用例类,在第一张UML图中也做了简要的说明。
ActivityInstrumentationTestCase2:主要是用于进行activity的功能测试,和activity的交互测试,如activity间的跳转,ui的交互等。

ActivityInstrumentationTestCase:这个类现在已deprecated了,所以不许考虑。

SingleLaunchActivityTestCase:该测试用例仅掉用setUp和tearDown一次,而不像其它测试用例类一样,没调用一次测试方法就会重新调用一次setUp和tearDown。所以主要测试activity是否能够正确处理多次调用。

ActivityUnitTestCase:主要用于测试Activity,因为它允许注入MockContext和MockApplicaton,所以可以测试Activity在不同资源和应用下的情况。

还有Application等的测试用例比较简单,看uml图。如果觉得不够详细,可以参考sdk文档的dev guide和api reference。

MainActivityTest测试用例
public class MainActivityTest extends ActivityInstrumentationTestCase2<MainActivity> {  
    private static final String TAG = "=== MainActivityTest";  
  
    private Instrumentation mInstrument;  
    private MainActivity mActivity;  
    private View mToLoginView;  
  
    public MainActivityTest() {  
        super("yuan.activity", MainActivity.class);  
    }  
  
    @Override  
    public void setUp() throws Exception {  
        super.setUp();  
        mInstrument = getInstrumentation();  
        // 启动被测试的Activity  
        mActivity = getActivity();  
        mToLoginView = mActivity.findViewById(yuan.activity.R.id.to_login);  
    }  
  
    public void testPreConditions() {  
        // 在执行测试之前,确保程序的重要对象已被初始化  
        assertTrue(mToLoginView != null);  
    }  
  
  
    //mInstrument.runOnMainSync(new Runnable() {  
    //  public void run() {  
    //      mToLoginView.requestFocus();  
    //      mToLoginView.performClick();  
    //  }  
    //});  
    @UiThreadTest  
    public void testToLogin() {  
        // @UiThreadTest注解使整个方法在UI线程上执行,等同于上面注解掉的代码  
        mToLoginView.requestFocus();  
        mToLoginView.performClick();  
    }  
  
    @Suppress  
    public void testNotCalled() {  
        // 使用了@Suppress注解的方法不会被测试  
        Log.i(TAG, "method 'testNotCalled' is called");  
    }  
  
    @Override  
    public void tearDown() throws Exception {  
        super.tearDown();  
    }  
}  



LoginActivityTest测试用例
public class LoginActivityTest extends ActivityInstrumentationTestCase2<LoginActivity> {  
    private static final String TAG = "=== LoginActivityTest";  
  
    private Instrumentation mInstrument;  
    private LoginActivity mActivity;  
    private EditText mUsernameView;  
    private EditText mPasswordView;  
    private View mSubmitView;  
    private View mResetView;  
  
    public LoginActivityTest() {  
        super("yuan.activity", LoginActivity.class);  
    }  
  
    @Override  
    public void setUp() throws Exception {  
        super.setUp();  
        /* 
         *  要向程序发送key事件的话,必须在getActivity之前调用该方法来关闭touch模式 
         * 否则key事件会被忽略 
         */  
        setActivityInitialTouchMode(false);  
  
        mInstrument = getInstrumentation();  
        mActivity = getActivity();  
        Log.i(TAG, "current activity: " + mActivity.getClass().getName());  
        mUsernameView = (EditText) mActivity.findViewById(yuan.activity.R.id.username);  
        mPasswordView = (EditText) mActivity.findViewById(yuan.activity.R.id.password);  
        mSubmitView = mActivity.findViewById(yuan.activity.R.id.submit);  
        mResetView = mActivity.findViewById(yuan.activity.R.id.reset);  
    }  
  
    public void testPreConditions() {  
        assertTrue(mUsernameView != null);  
        assertTrue(mPasswordView != null);  
        assertTrue(mSubmitView != null);  
        assertTrue(mResetView != null);  
    }  
  
    public void testInput() {  
        input();  
        assertEquals("yuan", mUsernameView.getText().toString());  
        assertEquals("1123", mPasswordView.getText().toString());  
    }  
  
    public void testSubmit() {  
        input();  
        mInstrument.runOnMainSync(new Runnable() {  
            public void run() {  
                mSubmitView.requestFocus();  
                mSubmitView.performClick();  
            }  
        });  
    }  
  
    public void testReset() {  
        input();  
        mInstrument.runOnMainSync(new Runnable() {  
            public void run() {  
                mResetView.requestFocus();  
                mResetView.performClick();  
            }  
        });  
        assertEquals("", mUsernameView.getText().toString());  
        assertEquals("", mPasswordView.getText().toString());  
    }  
  
    @Override  
    public void tearDown() throws Exception {  
        super.tearDown();  
    }  
  
    private void input() {  
        mActivity.runOnUiThread(new Runnable() {  
            public void run() {  
                mUsernameView.requestFocus();  
            }  
        });  
        // 因为测试用例运行在单独的线程上,这里最好要  
        // 同步application,等待其执行完后再运行  
        mInstrument.waitForIdleSync();  
        sendKeys(KeyEvent.KEYCODE_Y, KeyEvent.KEYCODE_U,  
                KeyEvent.KEYCODE_A, KeyEvent.KEYCODE_N);  
  
        // 效果同上面sendKeys之前的代码  
        mInstrument.runOnMainSync(new Runnable() {  
            public void run() {  
                mPasswordView.requestFocus();  
            }  
        });  
        sendKeys(KeyEvent.KEYCODE_1, KeyEvent.KEYCODE_1,  
                KeyEvent.KEYCODE_2, KeyEvent.KEYCODE_3);  
    }  
}  


HomeActivityTest测试用例
public class HomeActivityTest extends ActivityUnitTestCase<HomeActivity> {  
    private static final String TAG = "=== HomeActivityTest";  
  
    private static final String LOGIN_CONTENT = "username:yuan\npassword:1123";  
  
    private HomeActivity mHomeActivity;  
    private TextView mLoginContentView;  
  
    public HomeActivityTest() {  
        super(HomeActivity.class);  
    }  
  
    @Override  
    public void setUp() throws Exception {  
        super.setUp();  
        Intent intent = new Intent();  
        intent.putExtra(HomeActivity.EXTRA_USERNAME, "yuan");  
        intent.putExtra(HomeActivity.EXTRA_PASSWORD, "1123");  
        // HomeActivity有extra参数,所以我们需要以intent来启动它  
        mHomeActivity = launchActivityWithIntent("yuan.activity", HomeActivity.class, intent);  
        mLoginContentView = (TextView) mHomeActivity.findViewById(yuan.activity.R.id.login_content);  
    }  
  
    public void testLoginContent() {  
        assertEquals(LOGIN_CONTENT, mLoginContentView.getText().toString());  
    }  
  
    @Override  
    public void tearDown() throws Exception {  
        super.tearDown();  
    }  
}  



接下来是运行测试用例,首先我们需要把要测试的程序安装到模拟器或真机上





运行测试用例,查看运行结果





这里仅仅讲了使用eclipse来进行单元测试,当然也是可以在命令行中进行单元测试的,但既然有eclipse这种图形界面的工具,就不再折腾什么命令行了。
还有就是测试用例也可以直接创建在源程序中(即源代码和测试代码放在一个项目中),具体怎么做的话google一些吧,就是把测试时涉及的一些Manifest元素移到源码工程的Manifest中等
分享到:
评论

相关推荐

    android单元测试实例二

    在这个"android单元测试实例二"中,我们将探讨如何对一个简单的加法函数进行单元测试,以及如何在同一个项目中组织测试代码。 首先,我们要理解单元测试的基本概念。单元测试是对软件中的最小可测试单元进行检查和...

    android单元测试实例一

    通过这个实例,你可以深入理解Android单元测试的工作原理,学会如何编写和执行测试用例,以及如何利用工具和框架优化测试流程。记住,良好的测试实践不仅能提升代码质量,还能降低维护成本,提高开发效率。

    Android JUnit单元测试实例

    这篇内容将深入探讨Android JUnit单元测试的基本实例,帮助开发者理解如何有效地利用它来验证应用程序的功能。 首先,我们需要了解Android JUnit的基本结构。一个简单的Android JUnit测试类通常会继承自`androidx....

    Android单元测试Demo

    本Demo旨在提供一个基础的Android单元测试实践,通过实例化和运行测试用例,确保应用程序的各个组件(如Activity、Service、BroadcastReceiver等)在独立环境中正确执行。 单元测试通常使用JUnit框架进行,这是Java...

    Android unitTest 按键单元测试项目

    一、Android单元测试基础 单元测试是针对软件中的最小可测试单元进行验证,对于Android应用来说,这通常是指单个的Java或Kotlin类、方法或函数。通过单元测试,开发者可以在不依赖实际设备或模拟器的情况下,独立地...

    android Junit单元测试

    Android JUnit 单元测试 在 Android 应用开发中,单元测试是一个非常重要的步骤。单元测试可以帮助开发者快速地检测应用中的错误和问题,从而提高应用的质量和 stability。Android JUnit 是一个基于 JUnit 框架的...

    android 单元测试代码例子

    总结起来,Android单元测试通过`JUnit`框架和Android测试扩展,可以有效地验证代码逻辑,确保功能正确性,并促进代码的健壮性。通过创建和运行测试用例,开发者可以快速发现和修复问题,提高整体应用的质量。

    Android学习笔记之应用单元测试实例分析

    本篇学习笔记主要探讨了如何在Android应用中进行单元测试,通过实例分析了其实现原理和具体步骤。下面将详细阐述这些内容。 首先,要启用单元测试,我们需要在`AndroidManifest.xml`文件中进行相应的配置。在`...

    教你在Android Studio 中进行单元测试(源码)

    一、Android单元测试概述 1. 单元测试定义:单元测试是对软件中的最小可测试单元进行检查和验证,对于Android应用来说,这通常是一个方法或者一个类。 2. 目的:单元测试能够确保代码的独立性,帮助开发者快速定位...

    Android代码-android-unit-testing-tutorial

    Android单元测试: 首先,从是什么开始 代码和测试代码都在what子package下面 Android单元测试(三):JUnit单元测试框架的使用 代码和测试代码在junit子package下面 Android单元测试在蘑菇街支付金融部门的实践 代码和...

    Android编程之单元测试实例分析

    总结,Android单元测试通过JUnit框架配合Android测试支持库,为开发者提供了一种有效验证代码功能的方式。通过合理的测试设计和执行,可以极大地提高代码质量和项目的可靠性。在实际开发中,我们应该养成良好的测试...

    Android编程典型实例与项目开发案例 光盘 源码

    12. **测试**:单元测试、集成测试,以及 Espresso 测试框架的应用。 13. **Android组件化与模块化**:理解如何将应用拆分为可重用的模块,便于维护和扩展。 14. **Gradle构建系统**:熟悉Gradle的构建脚本,理解...

    最详细的Android开发教程实例

    14. **测试和调试**:涵盖单元测试、UI测试和 Espresso 测试框架的使用,以及Android Studio的调试工具。 15. **发布应用**:讲解如何打包APK,签名验证,以及将应用上传到Google Play Store的过程。 本教程的目的...

    Android人品测试器

    开发者需要使用Android Studio的内置工具进行单元测试、集成测试,确保应用在不同设备和Android版本上的兼容性。 总结来说,《Android人品测试器》的开发涵盖了Android应用的基本架构、UI设计、事件处理、数据计算...

    android studio基础实例代码完整可执行代码.zip

    13. **单元测试和UI测试**:如何编写和运行JUnit或Espresso测试,确保代码质量。 14. **Android模拟器与真机调试**:如何在Android Studio中配置和使用模拟器,以及连接真实设备进行调试。 15. **发布应用**:了解...

Global site tag (gtag.js) - Google Analytics