Visual Studio 2005 新增的测试功能使我们的开发工作更为快捷,代码质量在一定程度上有所提高。想必大家对类的单元测试已经比较熟悉,这里我们来看看它提供的另一项更为有趣的测试——用户界面自动化测试。事实上这个测试的本质仍是单元测试,单元测试的各种方法在用户界面测试中仍然可用。
我们以一个简单的 Windows Form 应用程序为例(在 Visual Studio 2005 Team System 下演示,其它版本亦可参考)。如图所示,这是一个一元二次方程的求解程序。
在 Visual Studio 2005 中加载这个程序的工程,切换到待测试窗体(Form1)的源代码视图。添加一个用户界面测试最简单的方法是在事件函数声明代码上右击鼠标,在快捷菜单中点击“Create Unit Tests”。在弹出的对话框中点选待测试的函数(特别是事件函数),确定。当然,手工添加一个单元测试项目也是可以的,但通过快捷菜单自动生成的代码中包含了一些测试辅助类和对象的声明,使用起来更加方便。
本例中,我们由 btnWorkOut_Click 创建单元测试,自动生成的代码如下:
[DeploymentItem("XandY.exe")]
[TestMethod()]
public void btnWorkOut_ClickTest()
{
Form1 target = new Form1();
MyTestProject.XandY_Form1Aclearcase/" target="_blank" >ccessor accessor = new MyTestProject.XandY_Form1Accessor(target);
object sender = null; // TODO: Initialize to an appropriate value
EventArgs e = null; // TODO: Initialize to an appropriate value
accessor.btnWorkOut_Click(sender, e);
Assert.Inconclusive("A method that does not return a value cannot be verified.");
}
熟悉单元测试的朋友应该明白 [TestMethod()] 属性的含义。在这个测试函数中,target 是待测试窗体的一个实例。sender 和 e 是各个事件函数必备的参数,一般情况下取 null 即可。最重要的是 accessor 对象,它是我们进行用户界面自动化测试的“总代理”。通过它可以访问到待测试窗体的各个方法、属性和控件,并可调用控件已声明的各个事件。由此,我们可以用 accessor 模拟使用者通过键盘、鼠标对用户界面的各项操作,同时即时获知程序中各个变量的状态。有兴趣的朋友可以读读 XandY_Form1Accessor 类的定义,了解这个“总代理”工作的奥秘。
我们对测试代码进行以下修改,以便对程序运行的正确性进行测试。
[DeploymentItem("XandY.exe")]
[TestMethod()]
public void btnWorkOut_ClickTest()
{
Form1 target = new Form1();
MyTestProject.XandY_Form1Accessor accessor = new MyTestProject.XandY_Form1Accessor(target);
object sender = null; // TODO: Initialize to an appropriate value
EventArgs e = null; // TODO: Initialize to an appropriate value
target.Show();
target.Refresh();
System.Threading.Thread.Sleep(1000);
accessor.textBox1.Text = "2";
accessor.textBox2.Text = "4";
accessor.textBox3.Text = "2";
target.Refresh();
System.Threading.Thread.Sleep(1000);
accessor.btnWorkOut_Click(sender, e);
target.Refresh();
Assert.IsTrue(accessor.d >= 0);
Assert.IsNotNull(accessor.xx);
Assert.IsNotNull(accessor.xy);
Console.WriteLine("X1 = " + accessor.textBox4.Text);
Console.WriteLine("X2 = " + accessor.textBox5.Text);
System.Threading.Thread.Sleep(1000);
target.Close();
}
以上代码中,我们通过 target.Show() 呈现窗体,通过 target.Refresh() 可以即时地刷新窗体内容在屏幕的显示,通过 System.Threading.Thread.Sleep(1000) 来延时。这几段代码都不是必需的,加入它们是为了便于程序反应和测试者观察,我们也完全可以在不显示用户界面的情况下更快地运行自动测试。至于用 Assert 来监控变量状态和用 Console 做输出,用过单元测试的朋友都应该了解。
此时,我们运行这个测试,就会在屏幕上先后看到窗体被加载,三个文本框被填入数据,计算结果出现在下面的文本框中,最后窗体被关闭。
再来看看测试结果的输出,符合我们的预期。