PHP单元测试与数据库测试

简介:

PHP单元测试与数据库测试


我总感觉 PHP 的开发者们并没有对 PHP 的质量有所追求,可能是因为 PHP 的机制问题吧,让大部分的开发者总以为浏览器访问就没有问题,所以很多时候,做 PHP 开发的,就没有单元测试的这些概念了。能不能有点追求?

我个人也是 PHP,但同时我也比较讨厌那些完事就算了的开发者,作为一个开发者,或者说是一个产品的经手人,就应该用心地去做好每个细节,一次比一次要更好。

但是做单元测试,质量检查,是需要一定的时间和人力投入的,但我敢保证地说,你花时间投入的,绝对不会是没用的,一定对你,对项目来说,是一个质的提升,只要你肯投入时间用心去做。

屁话说太多了,那接下来简单讲讲 phpunit 吧,官网。

因为我们习惯用 composer,所以我们也使用 composer 安装吧。

安装与配置

 
  1. $ composer require phpunit/phpunit -vvv 

安装完 phpunit,bin 执行脚本会创建在 vendor/bin 目录下,命名为 phpunit, 执行 php vendor/bin/phpunit 执行测试脚本

配置 bin 目录:

 
  1.  
  2.   "config": { 
  3.  
  4.     "bin""bin" 
  5.  
  6.  } 
  7.  
  8. }  

配置 bin 目录产生的目录,执行 php bin/phpunit 脚本开始测试。

phpunit 可以配置在当前执行路径添加一个配置文件 phpunit.xml.dist 或者 phpunit.xml,内容如下:

 
  1. <phpunit 
  2.          colors="true" 
  3.          bootstrap="./vendor/autoload.php" 
  4.         > 
  5.     <testsuites> 
  6.         <testsuite> 
  7.             <directory>dir1</directory> 
  8.         </testsuite> 
  9.         <testsuite> 
  10.             <directory>dir2</directory> 
  11.         </testsuite> 
  12.     </testsuites> 
  13. </phpunit>  

可以通过配置目录和初始化信息,让脚本自动执行对应的测试用例。

基础使用

使用 PHPUnit 创建我们的测试用例:

 
  1. <?php 
  2.  
  3. class DemoTest extends PHPUnit_Framework_TestCase 
  4.      public function testPushAndPop() 
  5.         { 
  6.             $stack = []; 
  7.             $this->assertEquals(0, count($stack)); 
  8.      
  9.             array_push($stack, 'foo'); 
  10.             $this->assertEquals('foo', $stack[count($stack)-1]); 
  11.             $this->assertEquals(1, count($stack)); 
  12.      
  13.             $this->assertEquals('foo', array_pop($stack)); 
  14.             $this->assertEquals(0, count($stack)); 
  15.         } 
  16. }  

类名需要以 *Test 结尾,继承 PHPUnit_Framework_TestCase。需要测试的方法需要一 test 开头,表明是一个测试方法。

一般常用测试无非就是 "断言",说白了,就是看看产生的结果是不是符合预期,如果是,那就证明,已经测试通过,否则,失败,说明逻辑处理,存在一定的差异,导致不符合预期。

更多的测试使用方法请看官网用例: PHPUnit

初始化

当我们的测试对象继承了 PHPUnit 后,初始化方法就需要使用它本身提供的 setUp 方法,代表类初始化,可以在初始化方法中初始化一些资源,或者加载。

数据库测试

除了以上基础的测试之外,关键一点应该在动态的数据,需要去测试吗,如果需要,那应该怎么去测试? 生产环境,也需要这样测试? 这个曾经困惑这我的问题,已经解开。

解答:

composer 中,有 --no-dev 选项,用来部署生产环境,避免测试环境的数据或者代码跑在了生产环境下。并且生产环境上数据库操作是没有很高权限的操作,要是有的话,你得回去面壁思考一下了。

dbunit 每次测试都重置数据,其实在生产环境下,就重置不了了,第一个是composer --no-dev 已经没有执行权利了,要是有,数据库已经不允许清空操作了。

要是生产环境不需要这些东西,那么应该怎么测试。其实需要有一个模拟生产环境的测试环境,去模拟生产环境测试,当所有测试都OK没有问题,那么就可以发布到生产环境上,要是严格一些,生产环境也是需要一轮测试。

 
  1. $ composer require phpunit/dbunit -vvv 

更多测试可看: 数据库测试

 
  1. <?php 
  2. class DBTest extends PHPUnit_Extensions_Database_TestCase 
  3.     /** 
  4.      * @return PHPUnit_Extensions_Database_DB_IDatabaseConnection 
  5.      */ 
  6.     public function getConnection() 
  7.     { 
  8.         $pdo = new PDO('mysql::dbname=test;host=127.0.0.1''user''pass'); 
  9.         return $this->createDefaultDBConnection($pdo, ':memory:'); 
  10.     } 
  11.  
  12.     /** 
  13.      * @return PHPUnit_Extensions_Database_DataSet_IDataSet 
  14.      */ 
  15.     public function getDataSet() 
  16.     { 
  17.         return $this->createFlatXMLDataSet(dirname(__FILE__).'/_files/guestbook-seed.xml'); 
  18.     } 
  19. }  

getConnection 方法是获取数据库连接,继承数据库测试后,必须实现的一个方法,并且需要返回 PHPUnit_Extensions_Database_DB_IDatabaseConnection 对象,可以仿照上述写法即可。

getDataSet 方法是数据集,在创建数据库测试的时候,自动填充,测试,和删除。他执行的流程是,每个测试用例,都会填充一次,以保证不会被其他测试用例影响。当当前测试用例测试完成后,会 truncate 掉填充的数据。

数据集支持挺多种方法,可以自定义数组,yml,xml,可以根据自己的使用习惯,自定义填充数据。数据集可看: 点我

执行脚本 php vendor/bin/phpunit

然后去对应查看自己的数据表,是否多了一些填充的数据呢?

抽象自己的数据库测试类

在很多情况下,我们的业务可谓是各种各样吧,倘若 phpunit 提供的数据库测试还不能满足或者不够方便的时候,就需要扩展自己的数据库测试,来达到自己想要的效果。

幸好,phpunit 提供了灵活的扩展操作(肯定啦,别人肯定不会像你这么傻,写死吧。哈哈),我们可以很容易地去实现自己的数据库测试类。

 
  1. <?php 
  2.  
  3. abstract class MyApp_Tests_DatabaseTestCase extends PHPUnit_Extensions_Database_TestCase 
  4.     // 只实例化 pdo 一次,供测试的清理和装载基境使用 
  5.     static private $pdo = null
  6.  
  7.     // 对于每个测试,只实例化 PHPUnit_Extensions_Database_DB_IDatabaseConnection 一次 
  8.     private $conn = null
  9.  
  10.     final public function getConnection() 
  11.     { 
  12.         if ($this->conn === null) { 
  13.             if (self::$pdo == null) { 
  14.                 self::$pdo = new PDO('mysql::dbname=test;host=127.0.0.1''user''pass'); 
  15.             } 
  16.             $this->conn = $this->createDefaultDBConnection(self::$pdo, ':memory:'); 
  17.         } 
  18.  
  19.         return $this->conn; 
  20.     } 
  21. }  

至今为止,完成了最基础和入门的单元测试和数据库测试,最终数据库无非就是查看数据增删改查是否和预期一样。所以,配置完数据库测试后,就可以走回第一步,编写你的测试用例,断言测试了。

恭喜你,你已经构建完自己的单元测试环境了。接下来需要做的是,提高易用性,测试覆盖率。我只能帮你到这里了,接下来的路,自己走吧。


作者:黄总

来源:51CTO

相关文章
|
3天前
|
测试技术
测试基础 Junit单元测试框架
测试基础 Junit单元测试框架
10 2
测试基础 Junit单元测试框架
|
11天前
|
安全 测试技术 Go
Golang深入浅出之-Go语言单元测试与基准测试:testing包详解
【4月更文挑战第27天】Go语言的`testing`包是单元测试和基准测试的核心,简化了测试流程并鼓励编写高质量测试代码。本文介绍了测试文件命名规范、常用断言方法,以及如何进行基准测试。同时,讨论了测试中常见的问题,如状态干扰、并发同步、依赖外部服务和测试覆盖率低,并提出了相应的避免策略,包括使用`t.Cleanup`、`t.Parallel()`、模拟对象和检查覆盖率。良好的测试实践能提升代码质量和项目稳定性。
16 1
|
11天前
|
PHP 数据库
DIY私人图床:使用CFimagehost源码自建无需数据库支持的PHP图片托管服务-2
DIY私人图床:使用CFimagehost源码自建无需数据库支持的PHP图片托管服务
|
11天前
|
存储 PHP Apache
DIY私人图床:使用CFimagehost源码自建无需数据库支持的PHP图片托管服务-1
DIY私人图床:使用CFimagehost源码自建无需数据库支持的PHP图片托管服务
|
11天前
|
监控 JavaScript 前端开发
【TypeScript技术专栏】TypeScript的单元测试与集成测试
【4月更文挑战第30天】本文讨论了在TypeScript项目中实施单元测试和集成测试的重要性。单元测试专注于验证单个函数、类或模块的行为,而集成测试关注不同组件的协作。选用合适的测试框架(如Jest、Mocha),配置测试环境,编写测试用例,并利用模拟和存根进行隔离是关键。集成测试则涉及组件间的交互,需定义测试范围,设置测试数据并解决可能出现的集成问题。将这些测试整合到CI/CD流程中,能确保代码质量和快速响应变化。
|
11天前
|
安全 关系型数据库 MySQL
【PHP开发专栏】PHP PDO与MySQLi操作数据库
【4月更文挑战第30天】PHP数据库交互的核心是PDO和MySQLi。PDO提供数据访问抽象层,支持多种数据库,强调安全和灵活性;MySQLi则专为MySQL设计,有面向对象和过程化接口。两者都支持预处理语句,防止SQL注入。选择取决于项目需求,如可移植性、特定功能和API偏好。了解其使用有助于编写安全高效的代码。
|
11天前
|
关系型数据库 MySQL PHP
【PHP 开发专栏】PHP 连接 MySQL 数据库的方法
【4月更文挑战第30天】本文介绍了 PHP 连接 MySQL 的两种主要方法:mysqli 和 PDO 扩展,包括连接、查询和处理结果的基本步骤。还讨论了连接参数设置、常见问题及解决方法,如连接失败、权限和字符集问题。此外,提到了高级技巧如使用连接池和缓存连接信息以优化性能。最后,通过实际案例分析了在用户登录系统和数据管理中的应用。
|
12天前
|
SQL 缓存 PHP
【PHP开发专栏】PHP数据库查询优化技巧
【4月更文挑战第29天】本文探讨了PHP数据库查询优化技巧,包括数据库设计的规范化与反规范化,合理使用索引,优化查询逻辑,以及避免SELECT *。在SQL查询优化中,利用EXPLAIN分析查询、优化JOIN操作和子查询,以及改进WHERE条件。PHP层面的优化涉及预处理语句、缓存查询结果、分页查询优化和异步处理。此外,还提到了高级技术如数据库分区、读写分离和分布式数据库。通过这些方法,开发者能提升查询效率,优化应用性能和用户体验。
|
12天前
|
SQL 关系型数据库 MySQL
【PHP开发专栏】PHP与数据库交互入门
【4月更文挑战第29天】本文介绍了PHP与数据库交互的基础,包括选择MySQL或PostgreSQL等关系型数据库,使用MySQLi或PDO扩展进行连接。示例展示了如何使用PHP连接数据库,如MySQLi的面向对象连接方式和PDO的多数据库支持。此外,还讲解了执行SQL查询(如SELECT、INSERT、UPDATE、DELETE)的操作,并强调了安全性与错误处理,如使用预处理语句防止SQL注入。通过学习,读者可掌握PHP操作数据库的基本技能。
|
14天前
|
IDE 测试技术 持续交付
【专栏】利用Python自动化测试与单元测试框架提升代码质量与效率
【4月更文挑战第27天】本文探讨了Python自动化测试与单元测试框架在提升代码质量与效率中的作用。Selenium、Appium用于Web和移动应用自动化测试,pytest提供强大、易扩展的测试支持。unittest是Python标准的单元测试框架,支持结构化测试用例和丰富的断言。实践中,应制定测试计划,编写高质量测试用例,实行持续集成与测试,并充分利用测试报告。这些工具和策略能有效保障代码质量和提升开发效率。