您好,登錄后才能下訂單哦!
這篇文章主要介紹了python怎么使用Hypothesis來自動化單元測試,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
Hypothesis 是一個 Python 庫,用于讓單元測試編寫起來更簡單,運行時功能更強大,可以在代碼中查找您不會想到的極端情況。它穩定,強大且易于添加到任何現有測試框架中。它的工作原理是讓您編寫斷言每種情況都應該正確的測試,而不僅僅是您偶然想到的那些。
典型的單元測試需要自己寫一些測試用例,然后編寫測試函數,通過一段代碼運行它,然后根據預期結果檢查結果。
Hypothesis 有所不同。它是基于屬性進行單元測試。它通過生成與您的規范匹配的任意數據并檢查在這種情況下程序是否仍然有效。如果找到了一個失敗的用例,它將采用該示例并將其測試用例范圍縮減縮減為一定尺寸,然后對其進行簡化,直到找到一個仍會導致問題的小得多的示例。然后將其保存,后續單元測試時仍會使用這些用例。
現在就讓我們看看怎么用吧。
可以通過 pip 安裝,也可以通過源代碼安裝[2],也可以安裝一些擴展[3],如下:
pip install hypothesis pip install hypothesis[pandas,django]
先寫一段代碼,保存在 mycode.py 中,功能是對字符串進行特定的編碼和解碼,內容如下:
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
對這段代碼進行單元測試,往往需要寫很多測試用例,現在我們使用 hypothesis 來自動為我們測試,編寫 test_mycode.py (文件名隨意),內容如下:
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()
可以看出,這里并沒有出現具體的測試用例,而是使用來 text 的策略,相當于 hypothesis 自動窮舉來可能的情況,也可以看出它很容易可其他測試框架集成,這里是 unittest。現在來運行一下看看效果:
(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 的異常。現在我們來修復這個 bug,然后把所有的測試用例 s 給打印出來,看看它用了哪些測試用例。
encode 函數加入以下代碼:
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)
再次執行:
(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ÝÀãiÎ\U000ce9e5\x0b' s='\U0005bc37\U0005bc37\U000537a1ÝÀãiÎ\U000ce9e5\x0b' s='\U0005bc37\U000537a1\U000537a1ÝÀãiÎ\U000ce9e5\x0b' s='À\U000537a1\U000537a1ÝÀãiÎ\U000ce9e5\x0b' s='\U000965e1\x12\x85&\U000f500aÄÃc' s='\n\U0004466c\x86Î\x07' s='Ê\U00063f1e\x01G\x88' s='ÚV\n' s='VV\n' s='\U0008debf湆è' s='\U0008debf湆è' s='\U0008debf湆' s='\U0008debf\U0008debf' s='\U0008debf\U0008debfó]½àq\x82#\U00015196\U0001c8beg' s='\U0008debfgó]½àq\x82#\U00015196\U0001c8beg' s='?' s='Î' s='Î\U00085b9e' s="Î8'?\U00057c38Ù;\x07\U000a5ea8Ò»=\U00091d5b~8?" s='\U000d6497Ý>' s='\U000e0f01' s='\U000e0f01Å0y¢KN®' s='\U000e0f01Å0y¢KN®' s='\U00050a06' s='Å\U000b98b3か\U000ba80aá`Ã-Êu\x8c\x90³FÔ"' s='\x8e\U0004612a\x83ç' s='\x8e' s='\x8e\x98\U000fb3e0\U0010d2b3\x10\x82\x94Ð渥' s='¥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´Ð\x93ü\x9eb&i\x1cÑ' s='\U000ceb6f' s='\U000ceb6f\xa0\x08' s='\U000ceb6f\xa0\x08' s='\U000ceb6f?\x08' s='\U000ceb6f?勻\U0007cc15\U000b2aaa×**' s='\U000ceb6f?勻' s='勻?勻' s='J\x14?ö' s='q)' s='q)' s='q\U00060931' s='q6' s='\U000e3441' s='\U000e3441\U00019958¯' s='\x13' s='\U000f34dbk' s='Kp&tÛà' s='\nö\x93' s='\n\n\x93' s='\U00019c8dѳ\U00056cbd\U000e3b2f\U00058d302' s='\x90=R\x8bß\x03' s='\x9a' s='\U000147e7' s='\U000147e7\x85\U0007a3ef' s='\U000147e7\U00050a070Â>' s='\U000a4089\x0eC+RÁ\x02\x97\x9cüÌïSS\U0006cbc5;ÿ~\x16\x019VÇ\U000a32fdQ÷\x15' s='ÞÚ¾\x19©Z®' s='?æ' s='\U000cd45a' s='\U000cd45a\U000e15cbÑ\x08J\ueb3eúß\x07I\x91\x9a\x18\x16Ç\x80\x1a' s='\x8f}º\x0eq\x0b' s='\x0e}º\x0eq\x0b' s="\U000e05a3&¶º[fõ\x8bÜR'ͼt\x97íW\x05\U000caea9\U0008fd74\U000e8f1c¹?dfƾ\x13" s='\x10\U000e12e2ù\U0006f96erý\U00014baf\x00\x95\U000dbc92É\U00081613µ\U0003b865Z\U0008cc3c' s='ú\U000b561f\x8fÎ' s='\tàÖ÷' s='à\x92©Ì\U000618fa\x92' s='\U000aaf94\x94\x84\U000cda69\U0005291a\U000a63deþ¿O\x8a>\U000b458bÊ.\U00086f07\x1a' s='\U0009754e?U_\xa0\x13PQ\x18º\x07\U0006c9c5.Á' s='\U00102456' s='³W?Õ' s='\x14\x1c' s='\x14' s='\x14\U00105bcd"\x10Ô\x99\U000a5032R\U00056c44V&÷>+\U000aaff2ñ®\U000d7570%ª!\U00032553´8x^«' s='\x00\U000e2ac4¼ÄUrB' s='\x00\U000e2ac4¼ÄUrB' s='\x00\U000e2ac4¼ÄUrB' s='ª\x1aU\x8aÇ\U000b2fb9\U0005a586' . ---------------------------------------------------------------------- Ran 1 test in 0.180s OK
從執行結果可以看出,'' 首先被測試,其次 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
這一點非常有用,提升了測試代碼的可讀性,可以用來告訴開發人員或者未來的自己,輸入的字符串必須要考慮 '' 的情形。
此外,執行單元測試,不一定要使用 unittest.main(),也可以這樣,是不是很方便:
if __name__ == "__main__": test_decode_inverts_encode()
感謝你能夠認真閱讀完這篇文章,希望小編分享的“python怎么使用Hypothesis來自動化單元測試”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業資訊頻道,更多相關知識等著你來學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。