Android测试驱动开发实践1

在正式进行Android测试驱动开发之前,不得不先提一下Android应用架构问题。在传统软件开发中,MVC架构得到了广泛的应用,然而在Android开发中,很少见应用采用了MVC架构(不要说Android及Widget全部采用的是MVC架构,那是系统的事,我们讲的是应用程序开发),究其原因可能是初期Android应用大多较为简单,没有采用的必要,而后期一直在沿用初期的习惯。但是遇到一些复杂的应用,例如同样的数据在多个Activity中显示,如果数据分散在多个Activity中,那么数据发生更新,很有可能出现数据不一致的情况,尤其是Android应用为节省带宽大量使用缓存的情况下,这种情况更加突出。我所经历的项目中,大部分项目都或多或少存在这个问题。因此,采用MVC架构开发Android应用程序也许是一个值得尝试的方法。所以,在这里我们也会采用MVC架构来做应用程序的开发,虽然我们的架构未必在理论上很完备,也不一定能解决实际应用中的所有问题,但是采用MVC架构的的确确可以提高程序的质量,同时也会使程序的可测试性大大增强。

言归正传,我们现在就开测试驱动开发。

首先建立一个Android工程,例如wkj,然后建立一个Android测试工程如WkjTest,这样就构成了一个Android测试开发的起点,下面就让我们正式开始Android测试驱动开发吧!

我们开始编写第一个测试用例,由于是空工程,我们首先要测试的是MainActivity正确显示出AppModel所提供的应用名称-维康街。什么?Android创建的工程显示的是Hello World,怎么会是维康街?而且AppModel是个什么东西?细心的读者的这些问题没错,这就是测试驱动开发的工作模式:先写出失败的测试程序,然后再通过编程来通过这些测试用例。

打开WkjTest工作,创建新类MainActivityTest,并指定该类继承自android.test.ActivityInstrumentationTestCase2,代码如下所示:

package com.bjcic.wkj.test;

import com.bjcic.wkj.MainActivity;

import android.app.Activity;
import android.test.ActivityInstrumentationTestCase2;
import android.widget.TextView;

public class MainActivityTest extends ActivityInstrumentationTestCase2 {

	public MainActivityTest() {
		super(MainActivity.class);
	}
	
	/**
	 * 每个测试用例开始运行前必须执行的代码
	 */
	@Override
	protected void setUp() throws Exception {
		super.setUp();
		setActivityInitialTouchMode(false);
		activity = getActivity();
	}
	
	/**
	 * 每个测试用例运行结束时必须执行的代码
	 */
	@Override
	protected void tearDown() throws Exception {
		super.tearDown();
	}
	
	/**
	 * 测试界面上标题文字为:维康街
	 */
	public void testTemp001() {
		titleTxtv = (TextView)activity.findViewById(com.bjcic.wkj.R.id.j_titleTxtv);
		assertEquals(activity.getString(com.bjcic.wkj.R.string.welcome_msg), titleTxtv.getText());
	}
	
	private Activity activity = null; 
	private TextView titleTxtv = null;
}

在WkjTest工程中,选中MainActivityTest类,点右键,选择Run As....然后选Android JUnit Test,在随后弹出的界面选择测试设备,运行程序后,在Junit视图会显示,执行了test001测试用例,并且该测试用例失败了,这正是我们目前所需要的结果。

另外,注意我们对测试用例的命名,testTemp001,表明是临时性的测试用例。采用这种方式命名的测试用例,表明是在开发过程中使用的临时测试用例,不是用于回归测试中的正式测试用例。

下面开始编写功能性代码,来使这个测试用例通过。

还记得在本文开头处提到的MVC架构吗?现在我们就开始来一步步实现这个MVC架构。我们将Activity当作视图View,View读取Model中数据并显示到界面上,testTemp001这是为了测试这一工作模式。

在Android中怎样来实现一个Model呢?实现Model当然有很多种方式,我在这里采用扩展Application类来当作Model,即定义AppModel并继承Application,这样做的好处是Model不用采用Singleton模式也可以实现单例模式了,而且可以很方便在Activity中获取到。在我们这个最简单的应用中,需要Model中存在welcom_msg信息,这时在MainActivity中才可以通过向Model请求数据并显示在界面中。

AppModel类如下所示:

package com.bjcic.wkj;

import android.app.Application;

public class AppModel extends Application {
	// 生命周期方法开始
	public void onCreate() {
		super.onCreate();
		welcomeMsg = getString(R.string.welcome_msg);
	}
	
	public void onTerminate() {
		super.onTerminate();
	}
	// 生命周期方法结束
	
	
	
	private String welcomeMsg = null;



	public String getWelcomeMsg() {
		return welcomeMsg;
	}

	public void setWelcomeMsg(String welcomeMsg) {
		this.welcomeMsg = welcomeMsg;
	}
}

在MainActivity中添加如下代码:

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		appModel = (AppModel)getApplication();
	}
	
	@Override
	public void onResume() {
		super.onResume();
		((TextView)findViewById(R.id.j_titleTxtv)).setText(appModel.getWelcomeMsg());
	}
	
	private AppModel appModel = null;

最后在Manifest文件中添加:

 

这时再回到WkjTest工程,选中MainActivityTest类,右键选择Run As .... =>Android JUnit Test,这时Junit视图将显示测试用例通过。就这样,我们的第一个测试驱动开发周期就结束了。



你可能感兴趣的:(Android开发,Android测试驱动开发实践)