Android 单元测试

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

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

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

上图的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视图上显示了,如下图:

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

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

 

2.Activity的测试:

 

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

   效果图如下:

   登录界面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());
    }
    
}

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

 

本文转自:http://blog.csdn.net/stevenhu_223/article/details/8298858

posted @ 2015-04-16 11:08  似水流云  阅读(434)  评论(0编辑  收藏  举报