Python單元測(cè)試實(shí)例詳解
本文實(shí)例講述了Python單元測(cè)試。分享給大家供大家參考,具體如下:
在Python中進(jìn)行單元測(cè)試需要用到自動(dòng)單元測(cè)試框架PyUnit,Python2.1及其以后的版本都將PyUnit作為一個(gè)標(biāo)準(zhǔn)模塊(即python的unittest模塊),如果你很out,那么你需要從PyUnit網(wǎng)站下載源碼安裝后才能使用。
一、Python單元測(cè)試范例
測(cè)試最基本的原理是比較預(yù)期結(jié)果是否與實(shí)際執(zhí)行結(jié)果相同,如果相同則測(cè)試成功,否則測(cè)試失敗。為了更好地理解自動(dòng)測(cè)試框架PyUnit,下面會(huì)以對(duì)Widget類(lèi)進(jìn)行測(cè)試為例說(shuō)明之:
#widget.py
#將要被測(cè)試的類(lèi)Widget
class Widget:
def __init__(self, size = (40, 40)):
self._size = size
def getSize(self):
return self._size
def resize(self, width, height):
if width < 0 or height < 0:
raise ValueError, "illegal size"
self._size = (width, height)
def dispose(self):
pass
二、測(cè)試用例TestCase
軟件測(cè)試中最基本的組成單元式測(cè)試用例(test case),PyUnit使用TestCase類(lèi)來(lái)表示測(cè)試用例,并要求所有用于執(zhí)行測(cè)試的類(lèi)都必須從該類(lèi)繼承。TestCase子類(lèi)實(shí)現(xiàn)的測(cè)試代碼應(yīng)該是自包含的(self contained),即測(cè)試用例既可以單獨(dú)運(yùn)行,也可以和其它測(cè)試用例構(gòu)成集合共同運(yùn)行。TestCase類(lèi)中常用的函數(shù)或方法有:
setUp:進(jìn)行測(cè)試前的初始化工作。
tearDown:執(zhí)行測(cè)試后的清除工作。
failedinfo:表示不成立打印信息faliedinfo,為可選參數(shù)。
self.assertEqual(value1, value2, failedinfo):會(huì)無(wú)條件的導(dǎo)致測(cè)試失敗,不推薦使用。
self.assertTrue(, failedinfo):斷言value1 == value2。
self.assertFalse(, failedinfo):斷言value為真。
self.assertRaises(ValueError, self.widget.resize, -1, -1):斷言肯定發(fā)生異常,如果沒(méi)發(fā)生異常,則為測(cè)試失敗。參數(shù)1為異常,參數(shù)2為拋出異常的調(diào)用對(duì)象,其余參數(shù)為傳遞給可調(diào)用對(duì)象的參數(shù)。
TestCase在PyUnit測(cè)試框架中被視為測(cè)試單元的運(yùn)行實(shí)體,Python程序員可以通過(guò)它派生自定義的測(cè)試過(guò)程與方法(測(cè)試單元),利用Command和Composite設(shè)計(jì)模式,多個(gè)TestCase還可以組合成測(cè)試用例集合。PyUnit測(cè)試框架在運(yùn)行一個(gè)測(cè)試用例時(shí),TestCase子類(lèi)定義的setUp()、runTest()和tearDown()方法被依次執(zhí)行,最簡(jiǎn)單的測(cè)試用例只需要覆蓋runTest()方法來(lái)執(zhí)行特定的測(cè)試代碼就可以了。
1、靜態(tài)方法
一個(gè)測(cè)試用例只對(duì)軟件模塊中一個(gè)方法進(jìn)行測(cè)試,采用覆蓋runTest()方法來(lái)構(gòu)造測(cè)試用例,這在PyUnit中稱(chēng)之為靜態(tài)方法,舉例說(shuō)明如下:
#static.py
from widget import Widget
import unittest
#執(zhí)行測(cè)試的類(lèi)
class WidgetTestCase(unittest.TestCase):
def runTest(self):
widget = Widget()
self.assertEqual(widget.getSize(), (40, 40))
#測(cè)試
if __name__ == "__main__":
testCase = WidgetTestCase()
testCase.runTest()
如果采用靜態(tài)方法,Python程序員就不得不為每個(gè)要測(cè)試的方法編寫(xiě)一個(gè)測(cè)試類(lèi),該類(lèi)通過(guò)覆蓋runTest()方法來(lái)執(zhí)行測(cè)試,并在每個(gè)測(cè)試類(lèi)中生成一個(gè)待測(cè)試的對(duì)象,這樣會(huì)非常繁瑣與笨拙。
2、動(dòng)態(tài)方法
鑒于靜態(tài)方法的缺陷,PyUnit提供了另一種高帥富的解決方法,即動(dòng)態(tài)方法,只編寫(xiě)一個(gè)測(cè)試類(lèi)來(lái)完成對(duì)整個(gè)軟件模塊的測(cè)試,這樣對(duì)象的初始化工作可以在setUp()方法中完成,而資源的釋放則可以在tearDown()方法中完成,舉例說(shuō)明如下:
#dynamic.py
from widget import Widget
import unittest
class WidgetTestCase(unittest.TestCase):
def setUp(self):
self.widget = Widget()
def tearDown(self):
self.widget.dispose()
self.widget = None
def testSize(self):
self.assertEqual(self.widget.getSize(), (40, 40))
def testResize(self):
self.widget.resize(100, 100)
self.assertEqual(self.widget.getSize(), (100, 100))
動(dòng)態(tài)方法不再覆蓋runTest()方法,而是為測(cè)試類(lèi)編寫(xiě)多個(gè)測(cè)試方法,按照慣例這些方法通常以test開(kāi)頭但這不是必須的,在創(chuàng)建TestCase子類(lèi)的實(shí)例時(shí)必須給出測(cè)試方法的名稱(chēng)來(lái)為PyUnit測(cè)試框架指明運(yùn)行該測(cè)試用例時(shí)應(yīng)該調(diào)用測(cè)試類(lèi)中的哪些方法,這通常會(huì)結(jié)合測(cè)試用例集TestSuite一起使用。
三、測(cè)試用例集TestSuite
完整的單元測(cè)試很少只執(zhí)行一個(gè)測(cè)試用例,開(kāi)發(fā)人員通常需要編寫(xiě)多個(gè)測(cè)試用例才能對(duì)某一軟件功能進(jìn)行比較完成的測(cè)試,這些相關(guān)的測(cè)試用例稱(chēng)為一個(gè)測(cè)試用例集,在PyUnit中是用TestSuite類(lèi)來(lái)表示的。PyUinit測(cè)試框架允許Python程序員在單元測(cè)試代碼中定義一個(gè)名為suite()的全局函數(shù),并將其作為整個(gè)單元測(cè)試的入口,PyUnit通過(guò)調(diào)用它來(lái)完成整個(gè)測(cè)試過(guò)程:
def suite():
suite = unittest.TestSuite()
suite.addTest(WidgetTestCase("testSize"))
suite.addTest(WidgetTestCase("testResize"))
return suite
也可以直接定義一個(gè)TestSuite的子類(lèi),并在其初始化方法__init__中完成所有測(cè)試用例的添加:
class WidgetTestSuite(unittest.TestSuite)
def __init__(self):
unittest.TestSuite.__init__(self, map(WidgetTestCase, ("testSize", "testResize")))
這樣只需要在suite()方法中返回該類(lèi)的一個(gè)實(shí)例就可以了:
def suite():
return WidgetTestSuite()
在PyUnit測(cè)試框架中,TestSuite類(lèi)可以看成是TestCase類(lèi)的一個(gè)容器,用來(lái)對(duì)多個(gè)測(cè)試用例進(jìn)行組織,這樣多個(gè)測(cè)試用例可以自動(dòng)在一次測(cè)試中全部完成。事實(shí)上,TestSuite除了可以包含TestCase外,也可以包含TestSuite,從而可以構(gòu)成一個(gè)更龐大的測(cè)試用例集:
suite1 = mysuite1.TheTestSuite() suite2 = mysuite2.TheTestSuite() alltests = unittest.TestSuite((suite1, suite2))
四、實(shí)施測(cè)試TestRunner
編寫(xiě)測(cè)試用例(TestCase)并將它們組織成測(cè)試用例集(TestSuite)的最終目的只有一個(gè):實(shí)施測(cè)試并獲得最終結(jié)果。PyUnit使用TestRunner類(lèi)作為測(cè)試用例的基本執(zhí)行環(huán)境,來(lái)驅(qū)動(dòng)整個(gè)單元測(cè)試過(guò)程。但是Python開(kāi)發(fā)人員在進(jìn)行單元測(cè)試時(shí)一般不直接使用TestRunner類(lèi),而是使用其子類(lèi)TextTestRunner來(lái)完成測(cè)試,并將測(cè)試結(jié)果以文本方式顯示出來(lái)。舉例說(shuō)明如下:
#text_runner.py
from widget import Widget
import unittest
#執(zhí)行測(cè)試的類(lèi)
class WidgetTestCase(unittest.TestCase):
def setUp(self):
self.widget = Widget()
def tearDown(self):
self.widget.dispose()
self.widget = None
def testSize(self):
self.assertEqual(self.widget.getSize(), (40, 40))
def testResize(self):
self.widget.resize(100, 100)
self.assertEqual(self.widget.getSize(), (100, 100))
#測(cè)試
if __name__ == "__main__":
#構(gòu)造測(cè)試集
suite = unittest.TestSuite()
suite.addTest(WidgetTestCase("testSize"))
suite.addTest(WidgetTestCase("testResize"))
#執(zhí)行測(cè)試
runner = unittest.TextTestRunner()
runner.run(suite)
使用如下命令執(zhí)行該單元測(cè)試:
$python text_runner.py

默認(rèn)情況下,TextTestRunner將結(jié)果輸出到sys.stdout/sys.stderr上,但是如果在創(chuàng)建TextTestRunner類(lèi)實(shí)例時(shí)將一個(gè)文件對(duì)象傳遞給了構(gòu)造函數(shù),則輸出結(jié)果將被重定向到該文件中。
五、大道至簡(jiǎn)main()
PyUnit模塊中定義了一個(gè)名為main的全局方法,使用它可以很方便地將一個(gè)單元測(cè)試模塊變成可以直接運(yùn)行的測(cè)試腳本,main()方法使用TestLoader類(lèi)來(lái)搜索所有包含在該模塊中的測(cè)試方法,并自動(dòng)執(zhí)行它們。如果Python程序員能夠按照約定(以test開(kāi)頭)來(lái)命令所有的測(cè)試方法,那么只需要在測(cè)試模塊的最后加入如下幾行代碼即可:
if __name__ == "__main__": unittest.main()
下面是利用main()方法來(lái)進(jìn)行測(cè)試的完整例子:
#main_runner.py
from widget import Widget
import unittest
#執(zhí)行測(cè)試的類(lèi)
class WidgetTestCase(unittest.TestCase):
def setUp(self):
self.widget = Widget()
def tearDown(self):
self.widget.dispose()
self.widget = None
def testSize(self):
self.assertEqual(self.widget.getSize(), (40, 40))
def testResize(self):
self.widget.resize(100, 100)
self.assertEqual(self.widget.getSize(), (100, 100))
#測(cè)試
if __name__ == "__main__":
unittest.main()
使用如下命令執(zhí)行上面的單元測(cè)試:
$python main_runner.py
如上這樣將執(zhí)行WidgetTestCase中的所有測(cè)試方法,但是如果只想執(zhí)行testSize()方法,則可以如下這般:
$python main_runner.py WidgetTestCase.testSize
如果在單元測(cè)試腳本中定義了TestSuite,還可以指定要運(yùn)行的測(cè)試集,使用-h參數(shù)可以查看運(yùn)行該腳本所有可能用到的參數(shù):
$python main_runner.py -h

需要注意的是:PyUnit的TestCase中如果有多個(gè)test_xxx,則默認(rèn)按照xxx的字母順序執(zhí)行測(cè)試用例函數(shù),如果test_xxx之間有依賴(lài)關(guān)系的話就會(huì)出錯(cuò),解決方法有二:1、解耦;2、編寫(xiě)xxx函數(shù)時(shí)人為地按字母順序。
當(dāng)然,如果你安裝了Python 2.7.2及以上版本,你還可以利用discover函數(shù)來(lái)自動(dòng)發(fā)現(xiàn)并執(zhí)行測(cè)試用例:
$python2.7 -m unittest discover
更多關(guān)于Python單元測(cè)試的資料可以參看這里,還有這里。
更多Python相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《Python入門(mén)與進(jìn)階經(jīng)典教程》、《Python字符串操作技巧匯總》、《Python列表(list)操作技巧總結(jié)》、《Python編碼操作技巧總結(jié)》、《Python數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Python函數(shù)使用技巧總結(jié)》及《Python文件與目錄操作技巧匯總》
希望本文所述對(duì)大家Python程序設(shè)計(jì)有所幫助。
- Python單元測(cè)試及unittest框架用法實(shí)例解析
- Python unittest單元測(cè)試openpyxl實(shí)現(xiàn)過(guò)程解析
- Python unittest單元測(cè)試框架及斷言方法
- Python單元測(cè)試與測(cè)試用例簡(jiǎn)析
- python3.6編寫(xiě)的單元測(cè)試示例
- Python Django框架單元測(cè)試之文件上傳測(cè)試示例
- Python unittest單元測(cè)試框架的使用
- Python單元測(cè)試簡(jiǎn)單示例
- Python Unittest自動(dòng)化單元測(cè)試框架詳解
- Python如何在單元測(cè)試中給對(duì)象打補(bǔ)丁
相關(guān)文章
Python如何使用k-means方法將列表中相似的句子歸類(lèi)
這篇文章主要介紹了Python如何使用k-means方法將列表中相似的句子聚為一類(lèi),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08
django與小程序?qū)崿F(xiàn)登錄驗(yàn)證功能的示例代碼
這篇文章主要介紹了django與小程序?qū)崿F(xiàn)登錄驗(yàn)證功能的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02
python gensim使用word2vec詞向量處理中文語(yǔ)料的方法
這篇文章主要介紹了python gensim使用word2vec詞向量處理中文語(yǔ)料的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07
Python數(shù)據(jù)類(lèi)型中的“冒號(hào)“[::]——分片與步長(zhǎng)操作示例
這篇文章主要介紹了Python數(shù)據(jù)類(lèi)型中的“冒號(hào)“[::]——分片與步長(zhǎng)操作,結(jié)合實(shí)例形式分析了Python基本數(shù)據(jù)類(lèi)型中的分片與步長(zhǎng)使用方法及相關(guān)操作技巧,需要的朋友可以參考下2018-01-01
學(xué)習(xí)createTrackbar的使用方法及步驟
這篇文章主要為大家介紹了學(xué)習(xí)createTrackbar的使用方法及步驟,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-10-10
python使用seaborn繪圖直方圖displot,密度圖,散點(diǎn)圖
這篇文章主要介紹了python使用seaborn繪圖直方圖displot,密度圖,散點(diǎn)圖,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-07-07
pywinauto自動(dòng)化測(cè)試使用經(jīng)驗(yàn)
本文主要介紹了pywinauto自動(dòng)化測(cè)試使用經(jīng)驗(yàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03
Android申請(qǐng)相機(jī)權(quán)限和讀寫(xiě)權(quán)限實(shí)例
大家好,本篇文章主要講的是Android申請(qǐng)相機(jī)權(quán)限和讀寫(xiě)權(quán)限實(shí)例,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下2022-02-02

