溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

python怎么使用Hypothesis來自動化單元測試

發(fā)布時間:2021-03-23 09:23:00 來源:億速云 閱讀:355 作者:小新 欄目:開發(fā)技術(shù)

這篇文章主要介紹了python怎么使用Hypothesis來自動化單元測試,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

Hypothesis 是一個 Python 庫,用于讓單元測試編寫起來更簡單,運行時功能更強大,可以在代碼中查找您不會想到的極端情況。它穩(wěn)定,強大且易于添加到任何現(xiàn)有測試框架中。它的工作原理是讓您編寫斷言每種情況都應該正確的測試,而不僅僅是您偶然想到的那些。

Hypothesis 的基礎(chǔ)知識

典型的單元測試需要自己寫一些測試用例,然后編寫測試函數(shù),通過一段代碼運行它,然后根據(jù)預期結(jié)果檢查結(jié)果。

Hypothesis 有所不同。它是基于屬性進行單元測試。它通過生成與您的規(guī)范匹配的任意數(shù)據(jù)并檢查在這種情況下程序是否仍然有效。如果找到了一個失敗的用例,它將采用該示例并將其測試用例范圍縮減縮減為一定尺寸,然后對其進行簡化,直到找到一個仍會導致問題的小得多的示例。然后將其保存,后續(xù)單元測試時仍會使用這些用例。

現(xiàn)在就讓我們看看怎么用吧。

Hypothesis 快速入門

1、安裝

可以通過 pip 安裝,也可以通過源代碼安裝[2],也可以安裝一些擴展[3],如下:

pip install hypothesis
pip install hypothesis[pandas,django]

2、使用

先寫一段代碼,保存在 mycode.py 中,功能是對字符串進行特定的編碼和解碼,內(nèi)容如下:

def encode(input_string):
 count = 1
 prev = ""
 lst = []
 for character in input_string:
  if character != prev:
   if prev:
    entry = (prev, count)
    lst.append(entry)
   count = 1
   prev = character
  else:
   count += 1
 entry = (character, count)
 lst.append(entry)
 return lst


def decode(lst):
 q = ""
 for character, count in lst:
  q += character * count
 return q

對這段代碼進行單元測試,往往需要寫很多測試用例,現(xiàn)在我們使用 hypothesis 來自動為我們測試,編寫 test_mycode.py (文件名隨意),內(nèi)容如下:

from hypothesis import given
from mycode import decode,encode
from hypothesis.strategies import text
import unittest


class TestEncoding(unittest.TestCase):
 @given(text())
 def test_decode_inverts_encode(self, s):
  self.assertEqual(decode(encode(s)), s)


if __name__ == "__main__":
 unittest.main()

可以看出,這里并沒有出現(xiàn)具體的測試用例,而是使用來 text 的策略,相當于 hypothesis 自動窮舉來可能的情況,也可以看出它很容易可其他測試框架集成,這里是 unittest。現(xiàn)在來運行一下看看效果:

(py38env) ? tmp python test_mycode.py
Falsifying example: test_decode_inverts_encode(
 self=<__main__.TestEncoding testMethod=test_decode_inverts_encode>, s='',
)
E
======================================================================
ERROR: test_decode_inverts_encode (__main__.TestEncoding)
----------------------------------------------------------------------
Traceback (most recent call last):
 File "test_mycode.py", line 9, in test_decode_inverts_encode
 def test_decode_inverts_encode(self, s):
 File "/Users/aaron/py38env/lib/python3.8/site-packages/hypothesis/core.py", line 1162, in wrapped_test
 raise the_error_hypothesis_found
 File "test_mycode.py", line 10, in test_decode_inverts_encode
 self.assertEqual(decode(encode(s)), s)
 File "/Users/aaron/tmp/mycode.py", line 14, in encode
 entry = (character, count)
UnboundLocalError: local variable 'character' referenced before assignment

----------------------------------------------------------------------
Ran 1 test in 0.048s

FAILED (errors=1)

這里測試出當字符串為 '' 的時候會拋出 UnboundLocalError 的異?!,F(xiàn)在我們來修復這個 bug,然后把所有的測試用例 s 給打印出來,看看它用了哪些測試用例。

encode 函數(shù)加入以下代碼:

if not input_string:
 return []

test_mycode.py 文件打印出測試用例:

@given(text())
def test_decode_inverts_encode(self, s):
 print(f"{s=}")
 self.assertEqual(decode(encode(s)), s)

再次執(zhí)行:

(py38env) ? tmp python test_mycode.py
s=''
s='1'
s='0'
s='0'
s='0'
s='ā'
s='\U000cf5e5'
s='0'
s=''
s='0'
s='0'
s='E'
s=")dù'\x18\U0003deb3¤jd"
s='\U0005bc37\x07\U000537a1&Yacute;&Agrave;&atilde;i&Icirc;\U000ce9e5\x0b'
s='\U0005bc37\U0005bc37\U000537a1&Yacute;&Agrave;&atilde;i&Icirc;\U000ce9e5\x0b'
s='\U0005bc37\U000537a1\U000537a1&Yacute;&Agrave;&atilde;i&Icirc;\U000ce9e5\x0b'
s='&Agrave;\U000537a1\U000537a1&Yacute;&Agrave;&atilde;i&Icirc;\U000ce9e5\x0b'
s='\U000965e1\x12\x85&\U000f500a&Auml;&Atilde;c'
s='\n\U0004466c\x86&Icirc;\x07'
s='&Ecirc;\U00063f1e\x01G\x88'
s='&Uacute;V\n'
s='VV\n'
s='\U0008debf湆è'
s='\U0008debf湆è'
s='\U0008debf湆'
s='\U0008debf\U0008debf'
s='\U0008debf\U0008debfó]&frac12;àq\x82#\U00015196\U0001c8beg'
s='\U0008debfgó]&frac12;àq\x82#\U00015196\U0001c8beg'
s='?'
s='&Icirc;'
s='&Icirc;\U00085b9e'
s="&Icirc;8'?\U00057c38&Ugrave;;\x07\U000a5ea8&Ograve;&raquo;=\U00091d5b~8?"
s='\U000d6497&Yacute;>'
s='\U000e0f01'
s='\U000e0f01&Aring;0y&cent;KN&reg;'
s='\U000e0f01&Aring;0y&cent;KN&reg;'
s='\U00050a06'
s='&Aring;\U000b98b3か\U000ba80aá`&Atilde;-&Ecirc;u\x8c\x90&sup3;F&Ocirc;"'
s='\x8e\U0004612a\x83&ccedil;'
s='\x8e'
s='\x8e\x98\U000fb3e0\U0010d2b3\x10\x82\x94&ETH;渥'
s='&yen;W'
s='p\U000e5a2aE·`ì'
s='\U000b80f8\x12\U000c2d54'
s='.\U000703de'
s='6\U00010ffa\U000f7994\x8e'
s='116\U000f7994\x8e'
s='1?6\U000f7994\x8e'
s='4?6\U000f7994\x8e'
s='4\x8e6\U000f7994\x8e'
s='0'
s='\U0006a564&acute;&ETH;\x93ü\x9eb&i\x1c&Ntilde;'
s='\U000ceb6f'
s='\U000ceb6f\xa0\x08'
s='\U000ceb6f\xa0\x08'
s='\U000ceb6f?\x08'
s='\U000ceb6f?勻\U0007cc15\U000b2aaa×**'
s='\U000ceb6f?勻'
s='勻?勻'
s='J\x14?&ouml;'
s='q)'
s='q)'
s='q\U00060931'
s='q6'
s='\U000e3441'
s='\U000e3441\U00019958&macr;'
s='\x13'
s='\U000f34dbk'
s='Kp&t&Ucirc;à'
s='\n&ouml;\x93'
s='\n\n\x93'
s='\U00019c8d&Ntilde;&sup3;\U00056cbd\U000e3b2f\U00058d302'
s='\x90=R\x8b&szlig;\x03'
s='\x9a'
s='\U000147e7'
s='\U000147e7\x85\U0007a3ef'
s='\U000147e7\U00050a070&Acirc;>'
s='\U000a4089\x0eC+R&Aacute;\x02\x97\x9cü&Igrave;&iuml;SS\U0006cbc5;&yuml;~\x16\x019V&Ccedil;\U000a32fdQ÷\x15'
s='&THORN;&Uacute;&frac34;\x19&copy;Z&reg;'
s='?&aelig;'
s='\U000cd45a'
s='\U000cd45a\U000e15cb&Ntilde;\x08J\ueb3eú&szlig;\x07I\x91\x9a\x18\x16&Ccedil;\x80\x1a'
s='\x8f}&ordm;\x0eq\x0b'
s='\x0e}&ordm;\x0eq\x0b'
s="\U000e05a3&&para;&ordm;[f&otilde;\x8b&Uuml;R'&Iacute;&frac14;t\x97íW\x05\U000caea9\U0008fd74\U000e8f1c&sup1;?df&AElig;&frac34;\x13"
s='\x10\U000e12e2ù\U0006f96er&yacute;\U00014baf\x00\x95\U000dbc92&Eacute;\U00081613&micro;\U0003b865Z\U0008cc3c'
s='ú\U000b561f\x8f&Icirc;'
s='\tà&Ouml;÷'
s='à\x92&copy;&Igrave;\U000618fa\x92'
s='\U000aaf94\x94\x84\U000cda69\U0005291a\U000a63de&thorn;&iquest;O\x8a>\U000b458b&Ecirc;.\U00086f07\x1a'
s='\U0009754e?U_\xa0\x13PQ\x18&ordm;\x07\U0006c9c5.&Aacute;'
s='\U00102456'
s='&sup3;W?&Otilde;'
s='\x14\x1c'
s='\x14'
s='\x14\U00105bcd"\x10&Ocirc;\x99\U000a5032R\U00056c44V&÷>+\U000aaff2&ntilde;&reg;\U000d7570%&ordf;!\U00032553&acute;8x^&laquo;'
s='\x00\U000e2ac4&frac14;&Auml;UrB'
s='\x00\U000e2ac4&frac14;&Auml;UrB'
s='\x00\U000e2ac4&frac14;&Auml;UrB'
s='&ordf;\x1aU\x8a&Ccedil;\U000b2fb9\U0005a586'
.
----------------------------------------------------------------------
Ran 1 test in 0.180s

OK

從執(zhí)行結(jié)果可以看出,'' 首先被測試,其次 hypothesis 使用了大量的極端測試用例,減輕了手寫的負擔,大大提升了效率。

雖然 hypothesis 具有自動記憶功能,你仍然可以顯式的指定某個測試用例一直被測試,而且這是推薦的做法,比如我想在每次的測試中都測試 '',可以這樣寫:

from hypothesis import given, example
from hypothesis.strategies import text


@given(text())
@example("")
def test_decode_inverts_encode(s):
 assert decode(encode(s)) == s

這一點非常有用,提升了測試代碼的可讀性,可以用來告訴開發(fā)人員或者未來的自己,輸入的字符串必須要考慮 '' 的情形。

此外,執(zhí)行單元測試,不一定要使用 unittest.main(),也可以這樣,是不是很方便:

if __name__ == "__main__":
 test_decode_inverts_encode()

感謝你能夠認真閱讀完這篇文章,希望小編分享的“python怎么使用Hypothesis來自動化單元測試”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,更多相關(guān)知識等著你來學習!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI