一个单元(Unit)是指一个可独立进行的工作,独立进行指的是这个工作不受前一次或接下来的工作的结果影响,简单的说,就是不与上下文(Context)发生关系。
如果是在Java程式中,具体来说一个单元可以是指一个方法(Method),这个方法不依赖於前一次运行的结果,也不牵涉到後一次的运行结果。
举例来说,下面这个程式的gcd()方法可视为一个单元:
package onlyfun.caterpillar;
public class MathTool {
public static int gcd(int num1, int num2) {
int r = 0;
while(num2 != 0) {
r = num1 % num2;
num1 = num2;
num2 = r;
}
return num1;
}
}
下面的gcd()方法不视为一个单元,要完成GCD的计算,您必须呼叫setNum1()、setNum2()与gcd()三个方法:
package onlyfun.caterpillar;
public class MathFoo {
private static int num1;
private static int num2;
public static void setNum1(int n) {
num1 = n;
}
public static void setNum2(int n) {
num2 = n;
}
public static int gcd() {
int r = 0;
while(num2 != 0) {
r = num1 % num2;
num1 = num2;
num2 = r;
}
return num1;
}
}
然而要完全使用一个方法来完成一个单元操作在实行上是有困难的,所以单元也可广义解释为数个方法的集合,这数个方法组合为一个单元操作,完成一个工作。
不过设计时仍优先考虑将一个公开的(public)方法要设计为单元,而尽量不用数个公开的方法来完成一件工作,以保持介面简洁与单元边界清晰。
将工作以一个单元进行设计,这可以使得单元可以重用,并且也使得单元可以进行测试,进而促进类别的可重用性。
单元测试(Unit Test)指的自然就是对每一个工作单元进行测试,了解其运行结果是否符合我们的预期,例如当您撰写完MathTool类别之後,您也许会这么作个小小的测试程式:
package onlyfun.caterpillar.test;
import onlyfun.caterpillar.MathTool;
public class UnitTestDemo {
public static void main(String[] args) {
if(MathTool.gcd(10, 5) == 5) {
System.out.println("GCD Test OK!");
}
else {
System.out.println("GCD Test Fail!");
}
}
}
这个动作是开发人员很常作的动作,然而您必须自行看著测试程式的输出结果来了解测试是否成功,另一方面,测试程式本身也是个程式,在更复杂的测试中,您也许会遇到测试程式本身出错,而导致无法验证结果的情况。
JUnit是个测试框架,藉由它所提供的工具,您可以减少撰写错误测试程式的机会,而另一方面,您可以有更好的方法来检验测试结果,而不是看著一长串输出的文字。