編寫基于Property-based的單元測試

編寫基于Property-based的單元測試

作為一個開發者,你可能認為你的職責就是編寫代碼從而完成需求。我不敢茍同,開發者的工作是通過軟件來解決現實需求,編寫代碼只是軟件開發的其中一個方面,編寫可靠的軟件和產出有價值的代碼更加重要。而TDD則是前輩通過經驗總結出的一套切實可行的軟件開發實踐,TDD旨在幫助開發者編寫高質量的代碼。
TDD的過程可以總結為以下幾個步驟:

  1. 先添加一個測試用例
  2. 執行測試,查看這個測試的失敗結果
  3. 對代碼做少量修改
  4. 再次執行測試,查看測試結果
  5. 對代碼進行重構,執行測試

單元測試的局限性

設想你要編寫一個加法功能,接受兩個數字,返回這兩個數字的和。讓我們來按照TDD的流程走一遍:
1.添加一個測試用例

[Fact]
public void Given3And1ShouldReturn4()
{var result = Add(3, 1);result.Should().Be(4);
}

2.執行代碼,發現測試并不能通過,因為我們還沒有實現add方法
3.對代碼做少量修改,讓測試通過

public int Add(int a, int b)
{if(a==3 && b ==1){return 4;}return 0;
}

4.繼續編寫測試

[Fact]
public void Given1And2ShouldReturn3()
{var result = Add(1, 2);result.Should().Be(3);
}

5.修改代碼讓測試通過

public int Add(int a, int b)
{if(a==3 && b ==1){return 4;}if (a == 1 && b == 2){return 3;}return 0;
}

至此為止,你一直在遵守TDD的步驟,測試全部變成了綠色,但是你始終沒有得到正確的Add實現。

哪里出了問題?你也許會覺得,咱們實現的Add方法有問題,我們故意犯了一些顯而易見的錯誤從而給TDD挑毛病。但是我任然可以反駁,他之所以看起來是顯而易見的錯誤是因為對兩個數字求和這樣的需求是每個人都明白的道理,所以你才覺得顯而易見,試想這是一個正式的場景,你也許真的就編寫了這樣的代碼從而讓兩個測試用例都能恰好通過。

如果說我們并不是故意編寫了這樣的代碼,那么單元測試和TDD這種實踐本身可能就有一些瑕疵。

換個角度來說,我們之所以沒有編寫出完整的業務邏輯,是因為單元測試是用例驅動的,而有限的測試用例漏掉了很多可能性。
如果我們對a和b分別取100個隨機值,Add方法都能夠通過,那么我們幾乎很難編寫出上面的Add實現。

[Fact]
public void WhenAddTwoNumberShouldGetSum()
{for (int i = 0; i < 100; i++){var a = GetRandomNumber();var b = GetRandomNumber();var result = Add(a, b);result.Should().Be(a + b);}
}

要想保證這樣的測試通過,你只能編寫出正確的Add實現:

public int Add(int a, int b)
{return a + b;
}

這個測試看起來不錯,通過產生大量隨機的輸入來驅動代碼實現,但是這個代碼存在一個致命的問題,測試代碼和被測試代碼使用了相同的業務邏輯。

//我們期望的數字是a + b
result.Should().Be(a + b);//而被測對象也是a + b
public int Add(int a, int b)
{return a + b;
}

如果a + b這個邏輯本身就有問題,但是因為你在測試代碼里重復了這一有問題的邏輯,實際上你的測試并沒有發現任何問題。

Property-based測試

如果你不在測試代碼里重復a + b這個邏輯,你如何通過這100個隨機輸入來斷言測試的準確性?什么樣的斷言能被用在這100個隨機輸入的測試用例中?
答案是斷言Add這一能力的屬性,某種能夠適用于所有測試用例的屬性。
舉個例子:a + b = b + a

[Fact]
public void A_Add_B_Should_EqualTo_B_Add_A()
{for (int i = 0; i < 100; i++){var a = GetRandomNumber();var b = GetRandomNumber();var result1 = Add(a, b);var result2 = Add(b, a);result1.Should().Be(result2);}}

這一特性正好是加法交換律,如果只是測試交換律還是不能夠保證Add方法的準確性,因為你可以把Add方法實現為a * b。
我們還可以斷言起結合律,即a + b + c = a + (b + c)

[Fact]
public void A_Add_B_Add_C_Should_EqualTo_B_Add_C_Add_A()
{for (int i = 0; i < 100; i++){var a = GetRandomNumber();var b = GetRandomNumber();var c = GetRandomNumber();var result1 = Add(Add(a, b), c);var result2 = Add(a, Add(b, c));result1.Should().Be(result2);}
}

如何實踐Property-based測試

所以什么是Property-based測試?從上面的分析能夠看出Property-based測試實際上提出了兩個策略來保證測試的有效性:

  1. 隨機產生輸入值,保證足夠多的測試用例
  2. 找出并斷言功能具有的普遍適應性的屬性

在.NET領域,FsCheck用來進行Property-based測試,Property-based是從Haskell移植過來的,幾乎所有的主流語言都有其移植版本。
下篇我們將介紹如何通過FsCheck來做Property-based測試。

轉載于:https://www.cnblogs.com/xiandnc/p/9770241.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/250517.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/250517.shtml
英文地址,請注明出處:http://en.pswp.cn/news/250517.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

樹鏈剖分+線段樹 單點修改 區間求和 模板

馬上要去西安打邀請賽了&#xff0c;存下板子 首先是vector存圖的&#xff1a; #include<bits/stdc.h> using namespace std; #define ll long long #define lson l,m,rt<<1 #define rson m1,r,rt<<1|1 #define mid int m (l r) >> 1 const int M …

koa --- seesion實現登錄鑒權

koa vue session 實現一個簡單的登錄邏輯 /login component/login-session.html <!DOCTYPE html><head><script src"https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script><script src"https://unpkg.com/axios/dist/axios.…

BZOJ2216: [Poi2011]Lightning Conductor

第一道此類的題&#xff0c;所以這是一篇假的博客&#xff0c;定理不會證明不理性 也不一定對 我是從這篇博客看的 很顯然是讓你求 p[i] max{a[j] sqrt(i - j)} - a[i] 就是 max{a[j] sqrt(|i - j|)} 這是一個 1D/1D 動態規劃 考慮對于絕對值的情況不好做&#xff0c;那就…

HNOI2018游記

HNOI2018游記 day 0 上午稍微寫了下題保持手感,然后看了一下套路,感覺不會的還是不會. 下午去劃水在湖面上被吹成傻逼... 感覺沒有聯賽前那么緊張了,應該是聯賽考掛了的原因吧.. day1 早上大概7:40就到了考場,和同學聊了一會兒天,看了看配置就進去了. 進去之后敲配置沒有一遍對…

Java 試題七

Java 試題七 1、java中有幾種類型的流&#xff1f;JDK為每種類型的流提供了一些抽象類以供繼承&#xff0c;請說出他們分別是哪些類&#xff1f; 答&#xff1a;字節流&#xff0c;字符流。 字節流繼承于InputStream、OutputStream&#xff0c; 字符流繼承于Reader、Writer…

flume快速入門及應用

? Flume 簡介? Flume 的安裝與配置? Fumne 部署   Flume 是 Cloudera 提供的一個高可用、 高可靠、 分布式的海量日志采集、 聚合和傳輸的系統。 Flume 支持定制各類數據源如 Avro、 Thrift、 Spooling 等。 同時 Flume提供對數據的簡單處理&#xff0c; 并將數據處理結果…

koa --- jwt實現最簡單的Token認證

HTML 有如下html: 先看代碼后挑重點來說明: <!DOCTYPE html><head><script src"https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script><script src"https://unpkg.com/axios/dist/axios.min.js"></script></…

python基礎之常用的高階函數

前言 高階函數指的是能接收函數作為參數的函數或類&#xff1b;python中有一些內置的高階函數&#xff0c;在某些場合使用可以提高代碼的效率&#xff0e; map() map函數可以把一個迭代對象轉換成另一個可迭代對象&#xff0c;不過在python3中&#xff0c;結果都是一個map對象&…

Java 試題八

Java 試題八 1、java中有幾種方法可以實現一個線程&#xff1f;用什么關鍵字修飾同步方法? stop()和suspend()方法為何不推薦使用&#xff1f; 答&#xff1a;有兩種實現方法&#xff0c;分別是繼承Thread類與實現Runnable接口&#xff1b;用synchronized關鍵字修飾同步方法…

bzoj2957 奧妙重重的線段樹

https://www.lydsy.com/JudgeOnline/problem.php?id2957 線段樹的query和update竟然還可以結合起來用&#xff01; 題意&#xff1a;小A的樓房外有一大片施工工地&#xff0c;工地上有N棟待建的樓房。每天&#xff0c;這片工地上的房子拆了又建、建了又拆。他經常無聊地看著窗…

koa --- 使用Github OAuth登錄

準備 登錄github選擇右上角的setting Developer settings -> OAuth Apps -> Register a new application 填入基本信息 點擊綠色的按鈕,可以看見 client_id 和 client secret 理清思路: 開始時,一個登錄的連接,點擊連接.后臺監聽登錄(/login)路由,然后重定向到github…

[數據結構] - ArrayList探究

一 概述 ArrayList可以理解為動態數組&#xff0c;與java的數組相比&#xff0c;它的容量能動態曾長&#xff0c;ArrayList是List接口的可變數組的實現&#xff0c;允許包括null值在內的所有元素。除了實現List接口外&#xff0c;此類還提供一些方法來操作內部用來存儲列表的數…

10.10考試題

voteplus 【問題描述】 R 君博客上有?個投票板塊&#xff0c;?家可以使?投票的?式來表達??對某些問題的贊成或反對的意見。 投票結果是公開的&#xff0c;但是 R 君會把這個結果化成?個最簡分數&#xff0c;如 1:2,4:3。 注意到同?個最簡分數可能代表了不同的總?數&am…

koa --- 跨域,解析POST參數、路由配置

目標 將開發中經常遇見的問題寫在這里方便查詢. 使用Koa創建一個簡單的服務器 const Koa require("koa"); const app new Koa(); app.listen(3000, () >{console.log("[server] Server is running at http://localhost:3000") })使用koa2-cors解決…

mysql數據庫常用操作

目前最流行的數據庫&#xff1a; oracle、mysql、sqlserver、db2、sqline --&#xff1a;單行注釋 #&#xff1a;也是單行注釋 /* 注釋內容*/&#xff1a;多行注釋 mysql -uroot -p密碼&#xff1a;登錄mysql service mysqld restart重啟mysql /etc/my.cnfmysql的配置文件 /var…

數碼相機控制點的自動定位檢校

為簡化控制場相機檢校中的人工量測控制點的繁瑣工作,提高相機檢校精度,本文提出一種方法:只需均勻量測少量控制點的像方坐標獲取相機檢校初始參數,便可通過動態模板匹配實現單影像相機檢校的控制點高精度自動定位檢校。實驗證明此方法檢校精度與人工量測檢校精度相近。 https:/…

Java 常用類

Java 常用類 字符串相關類 String類&#xff1a;構造字符串對象 常量對象&#xff1a;字符串常量對象是用雙引號括起的字符序列。 例如&#xff1a;”你好”、”12.97”、”boy”等。 字符串的字符使用Unicode字符編碼&#xff0c;一個字符占兩個字節 String類較常用構…

koa --- restful規范及其栗子

遵循Restful規范的簡單的栗子 前端代碼: <html><head><script src"https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script><script src"https://unpkg.com/element-ui/lib/index.js"></script><script src&qu…

軟工五:四則運算

題目要求 本次作業要求兩個人合作完成&#xff0c;駕駛員和導航員角色自定&#xff0c;鼓勵大家在工作期間角色隨時互換&#xff0c;這里會布置兩個題目&#xff0c;請各組成員根據自己的愛好任選一題。 題目一&#xff1a; 我們在剛開始上課的時候介紹過一個小學四則運算自動生…

Tomcat 配置Https

https://www.cnblogs.com/wanghaoyuhappy/p/5267702.html JDK1.8 keytool 生存證書 C:\keys\tomcat.keystore 1:證書生成 命令如下: keytool -genkey -alias tomcat -keypass 123456 -keyalg RSA -keysize 1024 -keystore C:/keys/tomcat.keytore -storepass 123456 keytool 使…