第三节 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的工程模板.本文 ...
随机推荐
- mac jdk profile 永久的配置
配置java_home 在MAC中设置JAVA_HOME环境变量 环境变量要再etc目录下的profile文件中配置,这样才是永久的配置. cd /etc vi profile 按 i 键进入写模式 ...
- Spring常用配置 Scope
Bean的Scope Scope描述的是Spring容器如何新建Bean的实例的.Spring的Scope有以下几种,通过@Scope注解来实现. 1.Singleton:一个Spring容器中 ...
- Linux内存管理 - slab分配器和kmalloc
本文目的在于分析Linux内存管理机制的slab分配器.内核版本为2.6.31.1. SLAB分配器 内核需要经常分配内存,我们在内核中最常用的分配内存的方式就是kmalloc了.前面讲过的伙伴系统只 ...
- centos安装django
1.如果默认安装的是python2.6,先升级至python2.7 参考:http://www.cnblogs.com/tiger2soft/p/5677843.html 2.安装pip 先下载get ...
- jeDate日期控件
http://www.jayui.com/jedate/ 这是日期控件官网,可以去里面下载使用 前台 <%@ Page Language="C#" AutoEvent ...
- 3.Mysql集群------Mycat分库分表
前言: 分库分表,在本节里是水平切分,就是多个数据库里包含的表是一模一样的. 只是把字段散列的分到不同的库中. 实践: 1.修改schema.xml 这里是在同一台服务器上建立了4个数据库db1,db ...
- java基础必备单词讲解 day two
variable 变量 count 统计 sum 总数 salary 薪水 Scanner 接收 import 导入 eclipse 日食 control 控制 shift 改变 alt 替换键 ha ...
- 记录表TABLE中 INDEX BY BINARY_INTEGER 的作用
type my_number_arr is table of number index by binary_integer; 其作用是,加了”index by binary_integer ”后,my ...
- Linux中用户与用户组管理
1.基础知识 Linux作为一种多用户的操作系统(服务器系统),允许多个用户同时登陆到系统上,并响应每个用户的请求. 任何需要使用操作系统的用户,都需要一个系统账号,账号分为:管理员账号与普通用户账号 ...
- VM虚拟机里的Ubuntu系统怎么设置屏幕分辨率
说白了就是安装VMWare tools工具,步骤如下: 1)在VMWare中启动ubuntu虚拟机 2)在VMWare中:右键单击启动虚拟机,点击[安装vmware tools] 3)在ubuntu中 ...