成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一日一技:如何正確為歷史遺留代碼補充單元測試?

開發 項目管理
如果你的項目是從一開始就寫單元測試,那么你寫起來應該輕松又愉快,因為單元測試會促使你的代碼自身變成可測試的代碼。

我們知道,在軟件工程中,單元測試是保證軟件質量的重要手段之一。一個優秀的代碼,單元測試的代碼量,經常會超過被測試的代碼本身。一個理想化的開發團隊,可能有三分之二的時間是在寫測試,剩下的三分之一時間才是寫業務代碼。

如果你的項目是從一開始就寫單元測試,那么你寫起來應該輕松又愉快,因為單元測試會促使你的代碼自身變成可測試的代碼。

但如果你接手了一個大項目,里面已經有幾十萬行代碼了,那么給這些代碼補單元測試會讓你知道什么叫做痛不欲生。你會發現有一些函數,它讓你不知道怎么寫測試代碼。但你又不能隨便修改代碼的結構,誰知道會引起什么連鎖反應?

我們來看一個例子:

圖片

我想測試的是business_code?里面,check_data_dup分別返回True或者False的時候,下面代碼的邏輯。也就是說,我只關心第18-27行的邏輯。這個時候不關心MySQL和Redis。但是每次測試都要從他們里面讀取數據,這樣就會導致測試代碼依賴外部環境。如果MySQL或者Redis掛了,那么測試代碼就會運行失敗。

而且,就算Redis和MySQL沒有故障,你怎么知道你的data_id和pk,在數據庫中對應的是什么數據?為了分別走到特定的分支,你還需要去檢測數據庫中特定數據的id。萬一是測試環境,別人修改了里面的數據,你的測試也可能會掛掉。

如果直接使用Pytest來寫測試案例,代碼是這樣的:

圖片

可以看到,我運行Pytest以后,成功了一個,失敗了一個。這里我模擬出數據庫中沒有數據能夠讓check_data_dup?走到返回True邏輯的情況。

難道為了讓單元測試進行下去,我還要去數據庫構造一條特定的數據?這只是單元測試,又不是集成測試。

為了解決這個問題,我們就可以使用mock模塊。這是Python自帶的一個模塊,可以動態替換函數。

它的寫法非常簡單:

圖片

我們只需要使用@mock.patch裝飾器,裝飾測試函數就可以了。這個裝飾器接收兩個參數,第一個參數是被模擬的函數的路徑,以點分割;第二個參數是你想讓它返回的值。

從上圖可以看到,test_runner.py?運行以后,原本在read_data_from_redis和read_data_from_mysql中打印的兩段文字都沒有打印,說明這兩個函數已經被動態替換了,他們內部的代碼不會運行。只會直接返回我們預設的這個返回值。這樣一來就跟數據庫解耦了。

注意,在上圖中,由于我們已經mock了check_data_dup?,因此read_data_from_redis和read_data_from_mysql?兩個函數隨便返回什么值都可以。如果你想順帶也測試一下check_data_dup,那么可以不mock它,如下圖所示。

圖片

在check_data_dup?函數的邏輯中,如果data?參數含有字符x?,并且user_id?是偶數,就返回True?,否則返回False?。我們通過mock兩個讀數據的函數,分別設置不同的返回值,就能滿足讓check_data_dup返回不同值的條件。

mock.path有一個小坑,一定要注意。我們來看看下面這個文件結構:

圖片

read_data_from_redis和read_data_from_mysql?兩個函數分布在了不同的文件里面。在runner.py?中導入并使用了他們。test_runner.py?中,我們使用@mock.patch對這兩個函數定義的路徑打補丁進行替換。可是替換了以后,運行Pytest,會發現這兩個函數竟然正常運行了。也就是說我們的替換失敗了。

之所以會出現這種情況,是因為我們要打補丁的并不是這兩個函數定義的地方,而是使用的地方。我們在runner.py中,分別使用如下兩個語句:

from mysql_util.SqlUtil import read_data_from_mysql
from controller.lib.redis.RedisUtil import read_data_from_redis

導入了這兩個函數,我們也是在runner.py?中使用他們的。因此,@mock.patch?的第一個參數,依然應該是runner.read_data_from_redis和runner.read_data_from_mysql。

正確的做法如下圖所示:

圖片

mock.patch?還有更多高級用法,例如替換類,替換實例方法等等。可以在unittest.mock中找到他。從Python 3.3開始,官方自帶了unittest.mock?,它跟直接import mock的效果是一樣的。

責任編輯:武曉燕 來源: 未聞Code
相關推薦

2024-07-30 08:16:18

Python代碼工具

2024-07-30 08:11:16

2021-04-12 21:19:01

PythonMakefile項目

2021-09-14 21:29:01

項目環境變量

2021-06-08 21:36:24

PyCharm爬蟲Scrapy

2021-10-15 21:08:31

PandasExcel對象

2025-05-28 03:15:00

Scrapy數據sleep

2020-12-04 06:39:25

爬蟲網頁

2023-10-29 09:16:49

代碼安全命令

2022-06-28 09:31:44

LinuxmacOS系統

2023-10-28 12:14:35

爬蟲JavaScriptObject

2022-03-12 20:38:14

網頁Python測試

2024-01-29 00:45:36

跨域后端接口

2024-11-11 00:38:13

Mypy靜態類型

2021-05-08 19:33:51

移除字符零寬

2021-11-12 05:00:43

裝飾器代碼功能

2021-04-27 22:15:02

Selenium瀏覽器爬蟲

2020-12-11 06:30:00

工具分組DataFrame

2024-05-29 00:00:01

字符串Python縮進

2022-03-07 09:14:04

Selenium鼠標元素
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品精品久久久久久 | 色偷偷噜噜噜亚洲男人 | 欧美日韩综合一区 | 欧美激情在线精品一区二区三区 | 在线婷婷 | 一区二区三区视频在线 | 免费一级黄色录像 | 精国产品一区二区三区 | 国产一级一级毛片 | 亚洲午夜电影 | 国产999精品久久久 日本视频一区二区三区 | 一级网站 | 精品一区二区在线观看 | 日日人人| 亚洲国产精品自拍 | 日韩精品二区 | 精品欧美一区二区久久久伦 | 久久aⅴ乱码一区二区三区 亚洲欧美综合精品另类天天更新 | 草久视频| 日本天天操 | 亚洲国产精品视频 | 久草.com| 黄视频在线网站 | 欧美激情精品久久久久久变态 | 99久久免费精品国产男女高不卡 | h漫在线观看 | 国产精品久久午夜夜伦鲁鲁 | 中文字幕亚洲无线 | www.亚洲国产精品 | 日本在线看片 | 成人在线观看免费观看 | 超碰天天 | 亚洲精品视频在线播放 | 亚洲精品永久免费 | 亚洲最大av网站 | 成人免费淫片aa视频免费 | 91精品国产自产精品男人的天堂 | 国产精品国产a | 国产精品黄色 | 日本免费一区二区三区四区 | 欧美精品91 |