国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術文章
文章詳情頁

Python如何在單元測試中給對象打補丁

瀏覽:5日期:2022-07-15 11:49:37

問題

你寫的單元測試中需要給指定的對象打補丁, 用來斷言它們在測試中的期望行為(比如,斷言被調用時的參數個數,訪問指定的屬性等)。

解決方案

unittest.mock.patch() 函數可被用來解決這個問題。 patch() 還可被用作一個裝飾器、上下文管理器或單獨使用,盡管并不常見。 例如,下面是一個將它當做裝飾器使用的例子:

from unittest.mock import patchimport example@patch(’example.func’)def test1(x, mock_func): example.func(x) # Uses patched example.func mock_func.assert_called_with(x)

它還可以被當做一個上下文管理器:

with patch(’example.func’) as mock_func: example.func(x) # Uses patched example.func mock_func.assert_called_with(x)

最后,你還可以手動的使用它打補丁:

p = patch(’example.func’)mock_func = p.start()example.func(x)mock_func.assert_called_with(x)p.stop()

如果可能的話,你能夠疊加裝飾器和上下文管理器來給多個對象打補丁。例如:

@patch(’example.func1’)@patch(’example.func2’)@patch(’example.func3’)def test1(mock1, mock2, mock3): ...def test2(): with patch(’example.patch1’) as mock1, patch(’example.patch2’) as mock2, patch(’example.patch3’) as mock3: ...

討論

patch() 接受一個已存在對象的全路徑名,將其替換為一個新的值。 原來的值會在裝飾器函數或上下文管理器完成后自動恢復回來。 默認情況下,所有值會被 MagicMock 實例替代。例如:

>>> x = 42>>> with patch(’__main__.x’):... print(x)...<MagicMock name=’x’ id=’4314230032’>>>> x42>>>

不過,你可以通過給 patch() 提供第二個參數來將值替換成任何你想要的:

>>> x42>>> with patch(’__main__.x’, ’patched_value’):... print(x)...patched_value>>> x42>>>

被用來作為替換值的 MagicMock 實例能夠模擬可調用對象和實例。 他們記錄對象的使用信息并允許你執行斷言檢查,例如:

>>> from unittest.mock import MagicMock>>> m = MagicMock(return_value = 10)>>> m(1, 2, debug=True)10>>> m.assert_called_with(1, 2, debug=True)>>> m.assert_called_with(1, 2)Traceback (most recent call last): File '<stdin>', line 1, in <module> File '.../unittest/mock.py', line 726, in assert_called_with raise AssertionError(msg)AssertionError: Expected call: mock(1, 2)Actual call: mock(1, 2, debug=True)>>>>>> m.upper.return_value = ’HELLO’>>> m.upper(’hello’)’HELLO’>>> assert m.upper.called>>> m.split.return_value = [’hello’, ’world’]>>> m.split(’hello world’)[’hello’, ’world’]>>> m.split.assert_called_with(’hello world’)>>>>>> m[’blah’]<MagicMock name=’mock.__getitem__()’ id=’4314412048’>>>> m.__getitem__.calledTrue>>> m.__getitem__.assert_called_with(’blah’)>>>

一般來講,這些操作會在一個單元測試中完成。例如,假設你已經有了像下面這樣的函數:

# example.pyfrom urllib.request import urlopenimport csvdef dowprices(): u = urlopen(’http://finance.yahoo.com/d/quotes.csv?s=@^DJI&f=sl1’) lines = (line.decode(’utf-8’) for line in u) rows = (row for row in csv.reader(lines) if len(row) == 2) prices = { name:float(price) for name, price in rows } return prices

正常來講,這個函數會使用 urlopen() 從Web上面獲取數據并解析它。 在單元測試中,你可以給它一個預先定義好的數據集。下面是使用補丁操作的例子:

import unittestfrom unittest.mock import patchimport ioimport examplesample_data = io.BytesIO(b’’’'IBM',91.1r'AA',13.25r'MSFT',27.72rr’’’)class Tests(unittest.TestCase): @patch(’example.urlopen’, return_value=sample_data) def test_dowprices(self, mock_urlopen): p = example.dowprices() self.assertTrue(mock_urlopen.called) self.assertEqual(p, {’IBM’: 91.1, ’AA’: 13.25, ’MSFT’ : 27.72})if __name__ == ’__main__’: unittest.main()

本例中,位于 example 模塊中的 urlopen() 函數被一個模擬對象替代, 該對象會返回一個包含測試數據的 ByteIO()

還有一點,在打補丁時我們使用了 example.urlopen 來代替 urllib.request.urlopen 。 當你創建補丁的時候,你必須使用它們在測試代碼中的名稱。 由于測試代碼使用了 from urllib.request import urlopen ,那么 dowprices() 函數 中使用的 urlopen() 函數實際上就位于 example 模塊了。

本節實際上只是對 unittest.mock 模塊的一次淺嘗輒止。 更多更高級的特性,請參考 官方文檔

以上就是Python如何在單元測試中給對象打補丁的詳細內容,更多關于Python 單元測試的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 日韩高清免费观看 | 一区二区三区视频免费观看 | 国产精品黄色 | 国产免费亚洲 | 美女扒开双腿让男人桶 | 碰碰碰人人澡人人爱摸 | 国产看片一区二区三区 | 亚洲悠悠色综合中文字幕 | 欧美亚洲精品在线 | 成人欧美视频免费看黄黄 | 亚洲美女视频网站 | 亚洲综合色一区二区三区小说 | 香蕉超级碰碰碰97视频在线观看 | 一区精品视频 | 91成人免费 | 久久老司机波多野结衣 | 琪琪午夜伦埋大全影院 | 欧美日本国产 | 日韩天天摸天天澡天天爽视频 | 免费播放巨茎人妖不卡片 | 国产在线精品一区二区中文 | 日本综合久久 | 一区二区三区在线 | 欧 | 亚洲欧美久久精品一区 | 成人网18免费下 | 国产美女又黄又爽又色视频免费 | 人人99| 加勒比日本道 | 视频三区精品中文字幕 | 99久久精品国产亚洲 | 国产精品自在自线 | www.av网| 日韩欧美国产亚洲 | 亚洲国产爱久久全部精品 | 久久视精品| 播播网手机在线播放 | 久草热视频在线观看 | 粉嫩高中生的第一次在线观看 | 2022国产91精品久久久久久 | 国产成人ay手机在线观看 | 国产亚洲一欧美一区二区三区 |