分类:
2006-02-21 15:14:40
调试程序是一个漫长的过程,程序越长越复杂,调试起来就愈加困难。如果你调试的是php程序,那么不妨采用phpUnit,它可以大大加快你的调试速度。 |
何谓PhpUnit |
Phpunit 脱胎于Fred Yankowski编写的著名的Junit测试框架。你可以到它的网站 下载最新的版本。你可以利用phpUnit编写一套测试软件包。保证你的程序代码正确无误。只需一步便可自动完成所有的测试。 |
如果监测到bug,你就可以再写一小段测试代码来找出错误之所在。日后若再有相同的bug出现,只要运行你先前的测试包,马上就可以抓到它。经常运行测试包便可以保证你的程序代码的强壮性。 |
开 始 |
假设我们有一个银行账务处理程序。现在需要为Account (账户) 类编写一个测试软件包。 |
以下是Account类 源代码: |
class Account{ |
var $balance; |
function Account($initialBalance=0){ |
$this->balance = $initialBalance; |
} |
function withdraw($amount){ |
$this->balance -= $amount; |
} |
function deposit($amount){ |
$this->balance += $amount; |
} |
function getBalance(){ |
return $this->balance; |
} |
function transferFrom(&$sourceAccount,$amount){ |
$sourceAccount->withdraw($amount); |
$this->deposit($amount); |
} |
?> |
创建一个测试类 |
首先,我们建立一个测试类AccountTest,它是一个由PhpUnit提供的TestCase的子类。在这个TestCase类中有2个基本的方 法:setUp和tearDown。 这2个方法的实现在父类中是空过程,必须由我们自己去重载。其中SetUp 用于进行AccountTest类的初始化处理。在本例中,我们对一些在测试中用到的账号进行初始化。tearDown 则用于AccountTest类的清空处理,在本例中无需使用。因此,就不对它进行重载。这样AccountTester类的源代码如下: |
class AccountTester extends TestCase{ |
var $_ac1; |
var $_ac2; |
var $_ac3; |
var $_ac4; |
function AccountTester($name){ |
$this->TestCase($name); // call parent constructor |
} |
function setUp(){ |
$this->_ac1 = new Account(100); // data for testWithdraw |
$this->_ac2 = new Account(20); // data for testDeposit |
$this->_ac3 = new Account(30); // data for testTransferFrom |
$this->_ac4 = new Account(50); |
} |
} |
?> |
加入专门的测试代码 |
现在,我们可以往向AccountTester类加入测试代码了。 |
// Make a withdrawal of 25 units from _ac1. |
// _ac1's initial balance is 100 |
function testWithdraw(){ |
$this->_ac1->withdraw(25); |
$this->assert($this->_ac1->getBalance() == 75); // 100 - 25 = 75 |
} |
// Make a deposit of 10 units into _ac2. |
// _ac1's initial balance is 20 |
function testDeposit(){ |
$this->_ac2->deposit(10); |
$this->assertEquals(30,$this->_ac2->getBalance()); //20 +10 = 30 |
} |
// Tranfers 10 units from _ac3 to _ac4 |
// _ac3's initial balance is 30 |
// _ac4's initial balance is 50 |
function testTransferFrom(){ |
$this->_ac4->transferFrom(&$this->_ac3,10); |
$this->assertEquals(20,$this->_ac3->getBalance(),"Source account balance incorrect"); // 30 - 10 = 20 |
$this->assertEquals(60,$this->_ac4->getBalance(),"Target account balance incorrect"); // 50 + 10 = 60 |
} |
?> |
这段代码中,assert(如同C里的断言)方法是测试的关键部分。如果在assert中的条件表达式为真,那么测试通过。否则返回错误。由于 assert方法大都用于判断两个变量的值是否相等。因此,testclass类引入了assertEquals方法专门实现这个功能。 AssertEquals方法中有3个参数,依次分别为:期望值,测试值,两值不相等时返回的消息提示串。 |
运行测试过程 |
好了,现在可以运行一下我们编好的测试程序了。我们还必须建立一个runtest.php测试程序来运行所有的测试过程。 |
runtest.php源代码如下: |
$tSuite = new TestSuite(); //creation of the test suite object 创建测试套件对象 |
$tSuite->addtest(new AccountTester("testWithdraw")); //Add inidividual tests |
$tSuite->addtest(new AccountTester("testDeposit")); //加入专门测试方法。 |
$tSuite->addtest(new AccountTester("testTransferFrom")); |
$res = new TextTestResult(); //Creation of the Result 建立一个测试结果类 |
$tSuite->run(&$res); //Run of the test 运行测试 |
$res->report(); //Print results 输出测试结果。 |
?> |
程序说明: |
首先创建测试套件对象tSuite,然后逐一加入专门测试方法,addtest方法的参数是测试方法的 再创建测试报告对象,随之运行测试。测试发现错误的结果由TestResult类捕捉,TestResult可以定制一套text/html的错误报告。 如果有必要你也可以自己编写输出部分。测试结果封装在TestResult类中,为了输出测试结果。我们采用了phpUnit提供的另外一个类 TextTestResult类,它可以输出文本或超文本格式的报告。当然我们也可以自己定制一个新的TestResult的子类控制更复杂的输出格式。 |
提示和技巧 |
1、在编写一个新的测试套件完成之后,我们可以先引入一个小小的bug以证明测试套件可以正常运行。 |
比如,在本例account类中,我们故意引入一个有问题的函数。 |
function withdraw($amount){ |
$this->balance -= $Amount; |
// 变量名大小写错误,本意是调用$amount参数,结果引入一个新变量$Amount。 |
} |
?> |
好,现在让我们运行测试套件,如果不出意外的话,我们将很快发现错误之处。 |
2.要指出的是,并非所有的方法都需要测试。你只需对相关的方法进行测试。 |
3.如果在开始编码前就写好测试代码,会使你更进一步明白你的程序到底需要完成什么样的任务。 |
现在,通过引入phpUnit的测试套件类,你可以发现找bug的时间缩短了,而你作为一个程序员的工作效率也提高了。 |
那么,尽情享受抓虫子的乐趣吧。祝您好胃口。 :))) |