廣州十大純?cè)O(shè)計(jì)公司seo難不難學(xué)
最近在看一本書《Test-Driven Development with Python》,里面非常詳細(xì)的介紹了如何一步一步通過測(cè)試驅(qū)動(dòng)開發(fā)(TDD)的方式開發(fā)Web項(xiàng)目。剛好這本書中使用了我之前所了解的一些技術(shù),Django、selenium、unittest等。所以,讀下來受益匪淺。
我相信不少開發(fā)都寫單元測(cè)試,不過,一般是先寫功能代碼,然后,再寫單元測(cè)試用例,在編寫單元測(cè)試用例的過程中,可能需要調(diào)整功能代碼,從而使單元測(cè)試用例通過。但是TDD就特別要求先寫測(cè)試用例,后寫實(shí)現(xiàn)代碼。這一開始確實(shí)有些難。
這里就選擇一個(gè)簡(jiǎn)單的例子向各位介紹一下TDD的流程(套路)。
編寫功能測(cè)試用例: ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ???
首先,編寫功能測(cè)試用例,functional_tests.py?
from selenium import webdriverbrowser = webdriver.Firefox() browser.get("http://127.0.0.1:8000")assert "Django" in browser.title
你沒看錯(cuò),這就是由Selenium編寫的功能測(cè)試代碼。打開Firefox瀏覽器,并訪問http://127.0.0.1:8000,通過assert 判斷瀏覽器標(biāo)題是否包含“Django”。
然后,運(yùn)行該測(cè)試用例。
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
測(cè)試用例失敗了,這是必然的,因?yàn)槲覀冞€沒有創(chuàng)建被測(cè)試的項(xiàng)目。但,這同樣也是我們想要的結(jié)果。TDD的套路就是通過編寫功能代碼,使測(cè)試用例通過。
創(chuàng)建項(xiàng)目: ? ? ? ? ? ? ? ?
接下來創(chuàng)建Django項(xiàng)目:
D:\pydj>django-admin startproject superlists
當(dāng)前項(xiàng)目結(jié)構(gòu)如下:?
├─ functional_tests.py
└─ superlists
├─ manage.py
└─ superlists
├─ __init__.py
├─ settings.py
├─ urls.py
└─ wsgi.py
進(jìn)入項(xiàng)目目錄,啟動(dòng)項(xiàng)目:
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.
再次運(yùn)行功能測(cè)試用例,functional_tests.py
接下來繼續(xù)編寫功能測(cè)試用例。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):# 小明聽說有一個(gè)很酷的在線代辦事項(xiàng)應(yīng)用# 她去看了這個(gè)應(yīng)用首頁(yè)self.browser.get("http://127.0.0.1:8000")# 它注意到網(wǎng)頁(yè)的標(biāo)題和頭部包含“To-Do”這個(gè)詞語(yǔ)。self.assertIn("To-Do", self.browser.title)if __name__ == '__main__':unittest.main()
這里用到了unittest?單元測(cè)試框架。如果,你不懂Python的單元測(cè)試,建議讀者去學(xué)習(xí)unittest。
執(zhí)行測(cè)試用例:
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)
測(cè)試用例又在預(yù)料之內(nèi)的失敗了!先不要著急解決這個(gè)問題,把項(xiàng)目創(chuàng)建完成。
D:\pydj\superlists>python3 manage.py startapp lists?
將functional_tests.py放到superlists項(xiàng)目目錄下。
單元測(cè)試與功能測(cè)試的區(qū)別: ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?
? 正如給事物所貼的眾多標(biāo)簽一樣,單元測(cè)試和功能測(cè)試之間的界線有時(shí)不那么清晰。不過,二者之間有個(gè)基本區(qū)別:功能測(cè)試站在用戶的角度從外部測(cè)試應(yīng)用,單元測(cè)試則站在程序員的角度從內(nèi)部測(cè)試應(yīng)用。
我遵從的 TDD 方法同時(shí)使用這兩種類型測(cè)試應(yīng)用。采用的工作流程大致如下。
(1) 先寫功能測(cè)試,從用戶的角度描述應(yīng)用的新功能。
(2) 功能測(cè)試失敗后,想辦法編寫代碼讓它通過(或者說至少讓當(dāng)前失敗的測(cè)試通過)。此時(shí),使用一個(gè)或多個(gè)單元測(cè)試定義希望代碼實(shí)現(xiàn)的效果,保證為應(yīng)用中的每一行代碼
(3) 單元測(cè)試失敗后,編寫最少量的應(yīng)用代碼,剛好讓單元測(cè)試通過。有時(shí),要在第 2 步和第 3 步之間多次往復(fù),直到我們覺得功能測(cè)試有一點(diǎn)進(jìn)展為止。
(4) 然后,再次運(yùn)行功能測(cè)試,看能否通過,或者有沒有進(jìn)展。這一步可能促使我們編寫一些新的單元測(cè)試和代碼等。
由此可以看出,這整個(gè)過程中,功能測(cè)試站在高層驅(qū)動(dòng)開發(fā),而單元測(cè)試則從低層驅(qū)動(dòng)我們做些什么。
打開/lists/tests.py文件,編寫單元測(cè)試。
from django.test import TestCase# Create your tests here.
class SmokeTest(TestCase):def test_bad_moths(self):self.assertEqual(1 + 1,2)
運(yùn)行單元測(cè)試:
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,說明單元測(cè)試沒問題。接下來就要編寫真正的單元測(cè)試了(/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>'))
第一個(gè)用例(test_root_url_resolves_to_home_page_view):
?resolve 是 Django 內(nèi)部使用的函數(shù),用于解析 URL,并將其映射到相應(yīng)的視圖函數(shù)上。檢查解析網(wǎng)站根路徑“ /” 時(shí),是否能找到名為 home_page 的函數(shù)。
第二個(gè)用例(test_home_page_returns_correct_html):
創(chuàng)建了一個(gè) HttpRequest 對(duì)象,用戶在瀏覽器中請(qǐng)求網(wǎng)頁(yè)時(shí), Django 看到的就是HttpRequest 對(duì)象。把這個(gè) HttpRequest 對(duì)象傳給 home_page 視圖,得到響應(yīng)。聽說響應(yīng)對(duì)象是 HttpResponse類的實(shí)例時(shí),你應(yīng)該不會(huì)覺得奇怪。接下來我們斷定響應(yīng)的 .content 屬性(即發(fā)送給用戶的 HTML)中有特定的內(nèi)容。
assertTrue()希望響應(yīng)以 <html> 標(biāo)簽開頭,并在結(jié)尾處關(guān)閉該標(biāo)簽。注意, response.content 是原始字節(jié),不是 Python 字符串,因此對(duì)比時(shí)要使用 b'' 句法。b是BYTE字符串
assertIn()希望響應(yīng)中有一個(gè) <title> 標(biāo)簽,其內(nèi)容包含單詞“ To-Do”——因?yàn)樵诠δ軠y(cè)試中做了這項(xiàng)測(cè)試。
在書中,這里的單元測(cè)試也不是一次寫成的,這里省略中間過程,一次寫成一個(gè)相對(duì)健全的單元測(cè)試。
根據(jù)單元測(cè)試編寫視圖文件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>')
運(yùn)行單元測(cè)試使其通過:
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òng)服務(wù):
D:\pydj\superlists>python manage.py runserver
運(yùn)行功能測(cè)試用例使其通過:
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)
最后感謝每一個(gè)認(rèn)真閱讀我文章的人,禮尚往來總是要有的,雖然不是什么很值錢的東西,如果你用得到的話可以直接拿走:
這些資料,對(duì)于【軟件測(cè)試】的朋友來說應(yīng)該是最全面最完整的備戰(zhàn)倉(cāng)庫(kù),這個(gè)倉(cāng)庫(kù)也陪伴上萬個(gè)測(cè)試工程師們走過最艱難的路程,希望也能幫助到你!?