第三节 MVC应用程序架构和测试
在查看如何测试单个功能之后,您可能会问,整个Web应用程序如何? 如前所述,有以下级别的测试:
- 单元测试
- 集成测试
- 功能测试
在开始编写测试时考虑这一点很重要。 可能还有其他类型的测试,但现在让我们关注这三种测试。 在谈论Web应用程序时,您将需要所有这些测试,但不同的测试有不同的应用场景。
您可能知道,MVC的设计模式被许多Web应用程序和框架使用。model是存储所有业务(主)逻辑的部分。 您绝对应该使用单元测试覆盖主要业务逻辑,可能没有数据库交互或API调用。 这有时是PHP应用程序的问题; 业务逻辑等于数据库访问,这并不总是正确的。 一些抽象并不是一个坏主意,诸如对象关系映射(ORM)之类的东西以及诸如Doctrine ORM或Propel之类的系统在测试方面确实很有帮助。
作为起点,您应该编写简单的PHPUnit测试而不进行数据库交互,只是为了涵盖主要业务逻辑(例如,增值税计算)。
接下来我们要详细探讨的是集成测试。 当您访问数据库或调用API时,最好知道您的代码如何与另一个系统交互,如果您的实现匹配,另一方面发生了什么。
与单元测试相比,集成测试可能会很慢 - 您可能无法像单元测试那样经常运行它们。 如果您正在使用数据库,那么这取决于您设置测试的方式。 如果您正在处理已知的数据集和数据库结构,那么数据库结构可能会在您不知情的情况下发生更改(然后,由于更改的数据库结构而未更新的代码可能是一个非常严重的问题,因此测试失败可能会很好)。 如果您直接使用第三方API,则永远不会知道它何时会发生故障或更改。 但同样,立即知道某些事情不太正确可能会很好。
控制器不应包含任何业务逻辑。 如果他们这样做,它通常是非常丑陋的意大利面条代码,其中代码是重复的,不一致的,甚至可能充满了错误。 控制器应该只处理(分派)请求并发送响应。 理论上,您可以为它们编写单元测试,但通常使用MVC框架提供的单元测试支持。 由于您可以启动整个应用程序,以便能够测试控制器功能,因此您需要进行功能测试。 执行数百行甚至数千行代码来测试简单的请求/响应。
最后但并非最不重要的是一种观点。 视图应该只处理和显示输出; 没有其他的。 为了保持严格的MVC结构并在前端和后端开发人员之间划分工作,最好使用模板系统(如Twig或Smarty)进行查看。 使用普通的PHP,做很多事情是非常诱人的。 甚至可以测试视图,但通常只需通过控制器或Selenium等工具进行功能测试,直接在浏览器中运行黑盒测试。
在谈到MVC设计模式时,结论应该是单元测试应该关注模型,并且应该应用更严格的MVC模式。 代码将具有更好的质量,更容易测试,并且可能包含比讨厌的意大利面条代码更少的错误,其中控制器是一切的主人。 但是你可以测试控制器,而现代框架通常都有帮助器来允许你测试控制器。
测试控制器
要了解如何测试控制器,让我们看看一些最着名的PHP MVC框架提供的内容。 这只是一个简短的概述; 您可能需要官方文档才能确切了解每个框架提供的内容。
Zend Framework 1的测试可以如下面的代码片段所示:
<?php class Zf1Test extends Zend_Test_PHPUnit_ControllerTestCase
{
public function setUp ()
{
$this->bootstrap = array ( $this, 'appBootstrap' );
parent::setUp();
} public function testIndexActionShouldContainLoginForm ()
{
$this->dispatch( '/' );
$this->assertAction( 'index' );
$this->assertResponseCode( 200 );
$this->assertQueryContentContains( 'h1', 'Hello World!' );
}
}
Zend Framework 2的测试可以显示为以下代码片段:
<?php
namespace ApplicationTest\Controller; use Zend\Test\PHPUnit\Controller\AbstractHttpControllerTestCase; class Zf2Test extends AbstractHttpControllerTestCase
{
public function setUp ()
{
$this->setApplicationConfig( include
'/path/to/application/config/test/
application.config.php' );
parent::setUp();
} public function testIndexActionCanBeAccessed ()
{
$this->dispatch( '/' );
$this->assertResponseStatusCode( 200 );
$this->assertModuleName( 'application' );
$this->assertControllerClass( 'IndexController' );
$this->assertActionName( 'index' );
$this->assertQueryContentContains( 'h1', 'Hello World!' );
}
}
Symphony 2的测试显示在以下代码段中:
<?php
namespace Application\Tests\Controller; use Symfony\Bundle\FrameworkBundle\Test\WebTestCase; class IndexControllerTest extends WebTestCase
{
public function testIndex ()
{
$client = static::createClient();
$crawler = $client->request( 'GET', '/' );
$this->assertTrue( $client->getResponse()->isSuccessful() );
$this->assertGreaterThan( 0,
$crawler->filter( 'html:contains
("Hello World!")' )->count() );
}
}
测试控制器非常有用,例如,在编写API时。 API可以非常容易地进行测试,应该进行测试。 但是以这种方式测试控制器可能非常昂贵,因为每次测试都必须一次又一次地启动整个框架和应用程序,这需要时间和资源。 您可能很想采用这种方法,因为在启动完整的应用程序时,您拥有所有可用资源,包括与数据库的连接。
第三节 MVC应用程序架构和测试的更多相关文章
- [译]MVC应用程序生命周期
原文:MVC Application Lifecycle 来一探究竟在MVC应用程序中参与请求处理的各个不同组件. 目录: 序言 背景 UrlRoutingModule RouteHandler Mv ...
- 使用MVC 5、Web API 2、KnockoutJS、Ninject和NUnit开发、架构和测试Web应用程序
做一名微软软件开发人员就像在国际煎饼屋订早餐一样.每道菜都有一堆煎饼,你必须从各种各样的煎饼和糖浆口味中选择.对于web应用程序,解决方案堆栈是一组软件子系统或组件,用于交付功能完整的解决方案(无论是 ...
- 转: GUI应用程序架构的十年变迁:MVC,MVP,MVVM,Unidirectional,Clean
十年前,Martin Fowler撰写了 GUI Architectures 一文,至今被奉为经典.本文所谈的所谓架构二字,核心即是对于对于富客户端的 代码组织/职责划分 .纵览这十年内的架构模式变迁 ...
- Android 程序架构: MVC、MVP、MVVM、Unidirectional、Clean...
摘选自:GUI 应用程序架构的十年变迁:MVC.MVP.MVVM.Unidirectional.Cleanhttps://zhuanlan.zhihu.com/p/26799645 MV* in An ...
- GUI应用程序架构的十年变迁:MVC,MVP,MVVM,Unidirectional,Clean
十年前,Martin Fowler撰写了 GUI Architectures 一文,至今被奉为经典.本文所谈的所谓架构二字,核心即是对于对于富客户端的 代码组织/职责划分 .纵览这十年内的架构模式变迁 ...
- 测试驱动 ASP.NET MVC 和构建可测试 ASP.NET MVC 应用程序
[测试驱动 ASP.NET MVC] http://t.cn/8kdi4Wl [构建可测试 ASP.NET MVC 应用程序]http://t.cn/8kdi4Wj
- [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序创建更复杂的数据模型
这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第六篇:为ASP.NET MVC应用程序 ...
- [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序实现继承
这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第十一篇:为ASP.NET MVC应用程 ...
- ASP.NET MVC应用程序执行过程分析
ASP.NET MVC应用程序执行过程分析 2009-08-14 17:57 朱先忠 朱先忠的博客 字号:T | T ASP.NET MVC框架提供了支持Visual Studio的工程模板.本文 ...
随机推荐
- java读取pfx或P12格式的个人交换库公私钥
使用的是CFCA签发的用于银行间交换数据的证书,下载后直接添加到浏览器中 1.导出 从浏览器导出p12文件(包含私钥) 2.验证 两种方式: openssl 代码(请注意alias别名是如何获取的): ...
- currency 过滤器
<!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...
- php红包算法函数[优化]
php红包算法 <?php header("Content-Type: text/html;charset=utf-8");//输出不乱码,你懂的 $total=10000; ...
- json_encode 的小技巧
做了一个 API 文档自动生成,解析的是每个 控制器类 的注释 json 数据,在做测试工具的时候,多层的 json 只有通过一个 textarea 把数据弄进去.如下图 怎么格式化 Json 数据并 ...
- yii2 的登录注册 轮子
//利用到了yii2 框架之中的验证规则 进行判定而已 也不是很高深的东西 但是 使用框架自身的轮子 会有安全性能的隐患 1注册reg controller 中 我都以admin 为例子 publi ...
- 如何删除hive表格的分区
今天的一个业务场景就是要把三年的数据从第一天不停的融合起来,每一天作为表格一个新的分区.由于空间有限,数据量很大,可能每天数据都是几十个G的大小.所以我需要做的一点就是在融合这一天之后,删除一天的分区 ...
- 第三章 文件 I/O
3.1 引言 先说明可用的文件 I/O 函数:open.read.write.close,然后说明不同缓冲区长度对read和write函数的影响. 本章所说的函数经常被称为不带缓冲的 I/O (unb ...
- python——int()函数
1. 使用 int() 将小数转换为整数,结果是向上取整还是向下取整呢? 小数取整会采用比较暴力的截断方式,即向下取整.(注:5.5向上取整为6,向下取整为5) 2. 我们人类思维是习惯于“四舍五入” ...
- Git-补丁文件交互
版本库间的交互是通过git push和/或git pull命令实现的,这是Git最主要的交互模式,但并不是全部.使用补丁文件是另外一种交互方式,适用于参与者众多的大型项目进行分布式开发. 创建补丁 G ...
- CC3200模块的内存地址划分和bootloader,启动流程(二)
1. 首先启动内部ROM固化的BOOT,然后这个ROM启动需要使用内存空间0X2000 0000 --- 0X2000 4000共16K的空间.一级BOOT的作用是串口升级和驱动库. 2. 然后是二级 ...