您可以对程序 PHP 进行单元测试,没问题。如果您的代码与 HTML 混合在一起,您肯定不会不走运。
在应用程序或验收测试级别,您的程序 PHP 可能依赖于超全局变量($_POST, $_GET, $_COOKIE 等)的值来确定行为,并通过包含模板文件并输出输出来结束。
要进行应用程序级测试,您只需设置超全局值即可;启动一个输出缓冲区(以防止一堆 html 淹没您的屏幕);调用页面;断言缓冲区内的东西;并在最后丢弃缓冲区。
所以,你可以这样做:
public function setUp()
{
if (isset($_POST['foo'])) {
unset($_POST['foo']);
}
}
public function testSomeKindOfAcceptanceTest()
{
$_POST['foo'] = 'bar';
ob_start();
include('fileToTest.php');
$output = ob_get_flush();
$this->assertContains($someExpectedString, $output);
}
即使对于包含大量包含的庞大“框架”,这种测试也会告诉您应用程序级功能是否正常工作。当您开始改进代码时,这将非常重要,因为即使您确信数据库连接器仍然可以工作并且看起来比以前更好,您会想要单击一个按钮并看到,是的,您仍然可以通过数据库登录和注销。
在较低级别,根据变量范围以及函数是否通过副作用(返回 true 或 false)工作,或者直接返回结果,会有细微的变化。
变量是否作为参数或函数之间的参数数组显式传递?还是变量设置在许多不同的地方,并作为全局变量隐式传递?如果它是(好的)显式案例,您可以通过以下方式对函数进行单元测试:(1)包括保存函数的文件,然后(2)直接提供函数测试值,以及(3)捕获输出并对其进行断言。如果您使用全局变量,您只需要格外小心(如上所述,在 $_POST 示例中)小心地将测试之间的所有全局变量清空。在处理推送和拉取大量全局变量的函数时,保持测试非常小(5-10 行,1-2 个断言)也特别有用。
另一个基本问题是函数是通过返回输出工作,还是通过改变传入的参数,返回真/假来代替。在第一种情况下,测试更容易,但同样,在两种情况下都有可能:
// assuming you required the file of interest at the top of the test file
public function testShouldConcatenateTwoStringsAndReturnResult()
{
$stringOne = 'foo';
$stringTwo = 'bar';
$expectedOutput = 'foobar';
$output = myCustomCatFunction($stringOne, $stringTwo);
$this->assertEquals($expectedOutput, $output);
}
在糟糕的情况下,您的代码会产生副作用并返回 true 或 false,您仍然可以很容易地进行测试:
/* suppose your cat function stupidly
* overwrites the first parameter
* with the result of concatenation,
* as an admittedly contrived example
*/
public function testShouldConcatenateTwoStringsAndReturnTrue()
{
$stringOne = 'foo';
$stringTwo = 'bar';
$expectedOutput = 'foobar';
$output = myCustomCatFunction($stringOne, $stringTwo);
$this->assertTrue($output);
$this->Equals($expectedOutput, $stringOne);
}
希望这会有所帮助。