通过测试驱动开发(TDD)的方式开发Web项目
最近在看一本书《Test-Driven Development with Python》,里面非常详细的介绍了如何一步一步通过测试驱动开发(TDD)的方式开发Web项目。刚好这本书中使用了我之前所了解的一些技术,Django、selenium、unittest等。所以,读下来受益匪浅。
我相信不少开发都写单元测试,不过,一般是先写功能代码,然后,再写单元测试用例,在编写单元测试用例的过程中,可能需要调整功能代码,从而使单元测试用例通过。但是TDD就特别要求先写测试用例,后写实现代码。这一开始确实有些难。
这里就选择一个简单的例子向各位介绍一下TDD的流程(套路)。
编写功能测试用例:
首先,编写功能测试用例,functional_tests.py
from selenium import webdriverbrowser = webdriver.Firefox()
browser.get("http://127.0.0.1:8000")assert "Django" in browser.title
你没看错,这就是由Selenium编写的功能测试代码。打开Firefox浏览器,并访问http://127.0.0.1:8000,通过assert 判断浏览器标题是否包含“Django”。
然后,运行该测试用例。
D:\pydj>python functional_tests.py
Traceback (most recent call last):
File "functional_tests.py", line 6, in <module>
assert "Django" in browser.title
AssertionError
测试用例失败了,这是必然的,因为我们还没有创建被测试的项目。但,这同样也是我们想要的结果。TDD的套路就是通过编写功能代码,使测试用例通过。
创建项目:
接下来创建Django项目:
D:\pydj>django-admin startproject superlists
当前项目结构如下:
├─ functional_tests.py
└─ superlists
├─ manage.py
└─ superlists
├─ __init__.py
├─ settings.py
├─ urls.py
└─ wsgi.py
进入项目目录,启动项目:
D:\pydj> cd superlists
D:\pydj\superlists>python manage.py runserver
Performing system checks...System check identified no issues (0 silenced).You have unapplied migrations; your app may not work properly until they are applied.
Run 'python manage.py migrate' to apply them.
June 13, 2016 - 23:23:29
Django version 1.9.7, using settings 'superlists.settings'
Starting development server at http://127.0.0.1:8000/
Quit the server with CTRL-BREAK.
再次运行功能测试用例,functional_tests.py

接下来继续编写功能测试用例。functional_tests.py
#coding=utf-8
from selenium import webdriver
import unittestclass NewVisitorTest(unittest.TestCase):def setUp(self):self.browser = webdriver.Firefox()self.browser.implicitly_wait(3)def tearDown(self):self.browser.close()def test_case_start_a_list_and_retrieve_it_later(self):# 小明听说有一个很酷的在线代办事项应用# 她去看了这个应用首页self.browser.get("http://127.0.0.1:8000")# 它注意到网页的标题和头部包含“To-Do”这个词语。self.assertIn("To-Do", self.browser.title)if __name__ == '__main__':unittest.main()
这里用到了unittest 单元测试框架。如果,你不懂Python的单元测试,建议读者去学习unittest。
执行测试用例:
C:\Python35\python.exe D:/pydj/functional_tests.py
F
======================================================================
FAIL: test_case_start_a_list_and_retrieve_it_later (__main__.NewVisitorTest)
----------------------------------------------------------------------
Traceback (most recent call last):File "D:/pydj/functional_tests.py", line 21, in test_case_start_a_list_and_retrieve_it_laterself.assertIn("To-Do", self.browser.title)
AssertionError: 'To-Do' not found in 'Welcome to Django'----------------------------------------------------------------------
Ran 1 test in 3.491sFAILED (failures=1)
测试用例又在预料之内的失败了!先不要着急解决这个问题,把项目创建完成。
D:\pydj\superlists>python3 manage.py startapp lists
将functional_tests.py放到superlists项目目录下。

单元测试与功能测试的区别:
正如给事物所贴的众多标签一样,单元测试和功能测试之间的界线有时不那么清晰。不过,二者之间有个基本区别:功能测试站在用户的角度从外部测试应用,单元测试则站在程序员的角度从内部测试应用。
我遵从的 TDD 方法同时使用这两种类型测试应用。采用的工作流程大致如下。
(1) 先写功能测试,从用户的角度描述应用的新功能。
(2) 功能测试失败后,想办法编写代码让它通过(或者说至少让当前失败的测试通过)。此时,使用一个或多个单元测试定义希望代码实现的效果,保证为应用中的每一行代码
(3) 单元测试失败后,编写最少量的应用代码,刚好让单元测试通过。有时,要在第 2 步和第 3 步之间多次往复,直到我们觉得功能测试有一点进展为止。
(4) 然后,再次运行功能测试,看能否通过,或者有没有进展。这一步可能促使我们编写一些新的单元测试和代码等。
由此可以看出,这整个过程中,功能测试站在高层驱动开发,而单元测试则从低层驱动我们做些什么。
打开/lists/tests.py文件,编写单元测试。
from django.test import TestCase# Create your tests here.
class SmokeTest(TestCase):def test_bad_moths(self):self.assertEqual(1 + 1,2)
运行单元测试:
D:\pydj\superlists>python3 manage.py test
Creating test database for alias 'default'...
.
----------------------------------------------------------------------
Ran 1 test in 0.001sOK
Destroying test database for alias 'default'...
OK,说明单元测试没问题。接下来就要编写真正的单元测试了(/lists/tests.py)。
from django.core.urlresolvers import resolve
from django.test import TestCase
from django.http import HttpRequestfrom lists.views import home_pageclass HomePageTest(TestCase):def test_root_url_resolves_to_home_page_view(self):found = resolve('/')self.assertEqual(found.func, home_page)def test_home_page_returns_correct_html(self):request = HttpRequest()response = home_page(request)self.assertTrue(response.content.startswith(b'<html>'))self.assertIn(b'<title>To-Do lists</title>', response.content)self.assertTrue(response.content.endswith(b'</html>'))
第一个用例(test_root_url_resolves_to_home_page_view):
resolve 是 Django 内部使用的函数,用于解析 URL,并将其映射到相应的视图函数上。检查解析网站根路径“ /” 时,是否能找到名为 home_page 的函数。
第二个用例(test_home_page_returns_correct_html):
创建了一个 HttpRequest 对象,用户在浏览器中请求网页时, Django 看到的就是HttpRequest 对象。把这个 HttpRequest 对象传给 home_page 视图,得到响应。听说响应对象是 HttpResponse类的实例时,你应该不会觉得奇怪。接下来我们断定响应的 .content 属性(即发送给用户的 HTML)中有特定的内容。
assertTrue()希望响应以 <html> 标签开头,并在结尾处关闭该标签。注意, response.content 是原始字节,不是 Python 字符串,因此对比时要使用 b'' 句法。b是BYTE字符串
assertIn()希望响应中有一个 <title> 标签,其内容包含单词“ To-Do”——因为在功能测试中做了这项测试。
在书中,这里的单元测试也不是一次写成的,这里省略中间过程,一次写成一个相对健全的单元测试。
根据单元测试编写视图文件lists/views.py
from django.shortcuts import render
from django.http import HttpResponse# Create your views here.
def home_page(request):return HttpResponse('<html><title>To-Do lists</title></html>')
运行单元测试使其通过:
D:pydj\superlists>python3 manage.py test
Creating test database for alias 'default'...
..
----------------------------------------------------------------------
Ran 2 tests in 0.002sOK
Destroying test database for alias 'default'...
最后不要忘了,配置superlists/urls.py文件。
urlpatterns = [
#url(r'^admin/', admin.site.urls),
url(r'^$', views.home_page),
]
最后的最后,启动服务:
D:\pydj\superlists>python manage.py runserver
运行功能测试用例使其通过:
C:\Users\fnngj\Desktop\superlists>python3 functional_tests.py
F
======================================================================
FAIL: test_can_start_a_list_and_retrieve_it_later (__main__.NewVisitorTest)
----------------------------------------------------------------------
Traceback (most recent call last):File "functional_tests.py", line 21, in test_can_start_a_list_and_retrieve_it_laterself.fail('Finish the test!')
AssertionError: Finish the test!----------------------------------------------------------------------
Ran 1 test in 7.070sFAILED (failures=1)
最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:
这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!

相关文章:
通过测试驱动开发(TDD)的方式开发Web项目
最近在看一本书《Test-Driven Development with Python》,里面非常详细的介绍了如何一步一步通过测试驱动开发(TDD)的方式开发Web项目。刚好这本书中使用了我之前所了解的一些技术,Django、selenium、unittest等。所以,读下来受益匪浅。 我相…...
技巧-PyCharm中Debug和Run对训练的影响和实验测试
简介 在训练深度学习模型时,使用PyCharm的Debug模式和Run模式对训练模型的耗时会有一些区别。 Debug模式:Debug模式在训练模型时,会对每一行代码进行监视,这使得CPU的利用率相对较高。由于需要逐步执行、断点调试、查看变量值等操…...
【古月居《ros入门21讲》学习笔记】07_创建工作空间和功能包
目录 说明: 1. 工作空间(workspace) 结构: 2. 创建工作空间和功能包 创建工作空间 编译工作空间 创建功能包 设置环境变量 3. 注意 同一个工作空间下,不能存在同名的功能包; 不同工作空间下,可以存在同名的功…...
第20章多线程
20.1线程简介 Windows操作系统是多任务操作系统,它以进程为单位。一个进程是一个包含有自身地址的程序,每个独立执行的程序都称为进程。也就是说每个正在执行的程序都是一个进程。系统可以分配给每一个进程有一段有限的使用CPU的时间(也可以称…...
深信服防火墙设置应用控制策略(菜鸟必看)
PS:前几天发布了关于深信服防火墙路由部署的流程:深信服防火墙路由模式开局部署-手把手教学(小白篇)-CSDN博客 昨天晚上有csdn的朋友联系我,说有一个关于ACL访问的问题要帮忙看一下 解决了以后,写个大概的…...
解锁 ElasticJob 云原生实践的难题
发生了什么 最近在逛 ElasticJob 官方社区时发现很多小伙伴都在头疼这个 ElasticJob 上云的问题,ElasticJob 本就号称分布式弹性任务调度框架,怎么在云原生环境就有了问题了呢,这就要从 Kubenertes 和 ElasticJob 的一些状态化说起。 有意思的…...
鸿蒙开发已成新趋势
随着华为鸿蒙操作系统的快速崭露头角,鸿蒙开发已然成为当前技术领域的热门新趋势。本文将深入探讨鸿蒙开发的重要性和独特优势,并详细介绍一些关键的鸿蒙开发技术和工具,以及它们对开发者个人和整个行业带来的深远影响。 首先,鸿蒙…...
万人拼团团购小程序源码系统+拼团设置+拼团管理 附带完整的搭建教程
随着互联网的快速发展,电子商务和社交电商的兴起,团购作为一种高效的营销策略和消费方式,受到了广大消费者的热烈欢迎。在此背景下,我们开发了一款基于微信小程序的万人拼团团购系统,旨在为用户提供一种更加便捷、高效…...
软信天成:速看!云端混合数据管理的最佳解决方案
智能时代,互联网、云计算和大数据的应用日益广泛,越来越多的企业将核心IT基础架构迁移至云上,以加速实现企业数字化转型,提高商业用户创新的可能性, 使 IT 变得更加灵活。 各个行业也正在进行从依赖本地系统到混合或云…...
GO 集成Prometheus
一、Prometheus介绍 Prometheus(普罗米修斯)是一套开源的监控&报警&时间序列数据库的组合,起始是由SoundCloud公司开发的。随着发展,越来越多公司和组织接受采用Prometheus,社会也十分活跃,他们便…...
ESP32-Web-Server 实战编程-通过网页控制设备的 GPIO
ESP32-Web-Server 实战编程-通过网页控制设备的 GPIO 概述 前述博客讲解了 Web 编程的基本知识,包括 HTML、CSS、JavaScript 三个部分,从这节开始,我们进入实战部分,在实际项目中进一步学习 ESP32-Web 编程。 GPIO (…...
Springboot 中 指定 AspectJ 的织入模式
在Spring Boot中,AspectJ的织入模式可以通过以下两种方式进行明确指定: 使用配置文件(application.properties或application.yml):在Spring Boot的配置文件中,可以添加以下属性来指定AspectJ的织入模式&am…...
【.NET全栈】.net的微软API接口与.NET框架源码
文章目录 0 前言1 微软官方.net接口学习2 .NET框架源码总结 0 前言 如果浏览器打不开链接,换一个浏览器打开。 我是 打不开微软的链接,使用: 可以打开!!! 1 微软官方.net接口学习 https://docs.microsoft…...
【深度学习】基于深度学习的超分辨率图像技术一览
超分辨率(Super-Resolution)即通过硬件或软件的方法提高原有图像的分辨率,图像超分辨率是计算机视觉和图像处理领域一个非常重要的研究问题,在医疗图像分析、生物特征识别、视频监控与安全等实际场景中有着广泛的应用。 SR取得了显著进步。一般可以将现有…...
Android12强制所有应用跟随gsensor旋转
前言 Android12系统中如果机器带gsensor,竖屏应用如果固定了竖屏,当机器旋转为横屏,竖屏应用是不会转到横屏显示的,还是竖屏显示。抖音这种app就是这样的。因为app里面manifest文件中通过android:screenOrientation固定住了竖屏显示。如果要让横屏的时候app也能够横屏显示,…...
C#常用运算符的优先级
前言 运算符在C#编程语言中扮演着重要的角色,用于执行各种计算和操作。了解运算符的优先级是编写高效和正确代码的关键。本文将深入探讨C#中38个常用运算符的优先级划分和理解,并提供详细的说明和示例,以帮助读者更好地理解运算符的使用。 目…...
鸿蒙4.0开发笔记之ArkTS语法的基础数据类型[DevEco Studio开发](七)
文章目录 一、基本数据类型的定义1、变量声明2、数字类型3、字符串类型4、布尔类型5、数组类型6、元组类型7、枚举类型8、联合类型(少用)9、未知Unkown类型10、未定义和空值类型 二、数据类型的使用1、组件内部声明变量不需要使用let关键字2、使用Divide…...
集成学习的两种常见策略:bagging VS. boosting
chatGPT回答,记在这里。 集成学习是一种通过组合多个弱学习器来构建一个更强大的学习器的方法。其中,bagging和boosting是两种常见的集成学习策略。 一、bagging & boosting 简介 Bagging(自助聚集法): Bagging…...
居家适老化设计第三十四条---卫生间之照明
居家适老化卫生间照明设计需要考虑以下几个方面:1. 光源选择:选择适合老年人眼睛的柔和光源,避免刺眼和眩光的发生。可以选择LED灯具,因为它们具有节能、寿命长和可调光的特点。2. 光线布置:在不同区域设置不同的光线&…...
如何使用Cloudreve将个人电脑打造为私有云盘并实现远程访问
文章目录 1、前言2、本地网站搭建2.1 环境使用2.2 支持组件选择2.3 网页安装2.4 测试和使用2.5 问题解决 3、本地网页发布3.1 cpolar云端设置3.2 cpolar本地设置 4、公网访问测试5、结语 1、前言 云存储概念兴起后,现在市面上也已经有了很多公有云盘。但一段时间后…...
本地化部署MT5:无需联网,保障敏感数据隐私的文本处理方案
本地化部署MT5:无需联网,保障敏感数据隐私的文本处理方案 1. 为什么选择本地化部署的文本处理方案 1.1 数据隐私保护的刚性需求 在当今数据驱动的商业环境中,企业面临着越来越严格的数据合规要求。许多行业如金融、医疗、法律等࿰…...
双模型协作方案:OpenClaw同时调用Qwen3-32B与Whisper实现会议转录
双模型协作方案:OpenClaw同时调用Qwen3-32B与Whisper实现会议转录 1. 为什么需要双模型协作 去年参加技术沙龙时,我注意到一个有趣现象:现场速记员总是两人一组工作。一人负责快速记录发言内容,另一人同步整理关键要点。这种分工…...
Qwen3.5-9B惊艳效果:上传Excel截图→识别表格→生成SQL查询语句演示
Qwen3.5-9B惊艳效果:上传Excel截图→识别表格→生成SQL查询语句演示 1. 开篇:认识Qwen3.5-9B的强大能力 Qwen3.5-9B是一款拥有90亿参数的开源大语言模型,它在多个领域展现出惊人的能力。这个模型最吸引人的特点是它不仅能处理文字ÿ…...
同样是硅做的圆片,价差百倍的秘密:半导体与光伏晶圆的本质区别
如果你留意过半导体和光伏行业的产品价格,一定会有这样的疑问:同样是硅材质、尺寸相近的晶圆,高端半导体晶圆一片能卖到几千元,而光伏硅片却只要几十元,身价相差近百倍。明明都是“硅做的圆片”,为何命运如此不同? 其实答案很简单:它们看似同宗同源,实则从一开始就被…...
可解释AI(XAI):让黑盒模型变得透明
XAI在软件测试中的革命性意义在人工智能(AI)技术迅猛发展的今天,深度学习等黑盒模型已成为软件系统的核心组件,广泛应用于推荐系统、自动驾驶、金融风控等领域。然而,这些模型的决策过程往往像“黑箱”一样不可预测&am…...
第九章:Vite API 参考手册
第九章:Vite API 参考手册 9.1 核心 API createServer 创建 Vite 开发服务器 import { createServer } from viteconst server await createServer({configFile: false,root: __dirname,server: { port: 3000 } })await server.listen() console.log(服务器启动在,…...
如何快速上手nomacs:10个必备技巧让图像浏览更高效
如何快速上手nomacs:10个必备技巧让图像浏览更高效 【免费下载链接】nomacs nomacs is a free image viewer for windows, linux, and mac systems. 项目地址: https://gitcode.com/gh_mirrors/no/nomacs nomacs是一款免费开源的图像查看器,支持W…...
2001-2023年各省农产品进出口额数据(无缺失)
2001-2023年各省农产品进出口额数据 1、时间:2001-2023年 2、来源:商务BU、农业年鉴、海关总署 3、指标:年份、省份、农产品出口额(亿元)、农产品进口额(亿元)、农产品出口额(万美…...
【PHP 8.9命名空间终极指南】:5大突破性增强、3个迁移避坑清单与向后兼容性权威验证
第一章:PHP 8.9命名空间增强的演进背景与核心定位PHP 命名空间自 5.3 版本引入以来,已成为组织大型代码库的事实标准。然而,随着现代 PHP 应用向模块化、跨域共享和静态分析深度依赖方向演进,原有命名空间机制在别名解析、嵌套声明…...
OpenClaw排错大全:Qwen3-4B接口超时与网关崩溃解决方案
OpenClaw排错大全:Qwen3-4B接口超时与网关崩溃解决方案 1. 问题背景与诊断准备 上周我在本地部署OpenClaw对接Qwen3-4B模型时,遭遇了典型的"三连击"问题:模型响应超时、网关频繁崩溃、飞书消息推送失败。经过72小时的折腾&#x…...

