[架構][技巧] AI UnitTest:讓ChatGPT幫助你編寫更好的測試代碼

閱讀時間約 25 分鐘
在軟體開發領域,AI工具的演進提供新方法來編寫和測試代碼。利用AI來輔助撰寫單元測試已經成為一種新的測試策略。
透過提示AI來撰寫測試代碼,可以更快速地生成測試用例,找出潛在的缺陷並解決問題,從而提高軟體開發的效率和品質。

流程

首先對ChatGPT描述使用的技術架構和想進行的步驟,接著在步驟中和ChatGPT互動並請它提供合適的單元測試代碼。(詳細內容請參考下方的”完整流程”)

優點

採用AI輔助撰寫單元測試的效果比預計還好,特別是測試範圍和效率
前者你可以明確指定AI提供測試的範圍,也可以要求AI提供容易遺漏的測試用例,擴大合理的涵蓋範圍。
後者比起手動輸入測試代碼,透過AI產生測試代碼的速度也會有顯著的提升。
提升速度這一點相當重要,許多排斥撰寫單元測試的開發者,主要原因為在開發時程中很難再排入撰寫單元測試的時間,更不用說當業務邏輯變動之後,與其對應的單元測試也必須重構修改,因此拒絕學習或撰寫單元測試。
若透過AI產生或重構測試代碼,就能大幅降低花費撰寫單元測試的時間,也能順便開啟通往TDD的大道。

缺點

不洩漏敏感訊息的情況下,將想測試的邏輯完整表達給ChatGPT是一個可能的問題,因為我測試的邏輯很簡單,ChatGPT很清楚溫度轉換公式,提供測試代碼相當容易,但若你想測試的演算法或情境特別複雜,要完整的表達清楚就沒這麼簡單了。

總結

在現階段透過AI來輔助撰寫單元測試是一種可以探索的方式,事實上整個過程就是簡單的TDD開發流程,而ChatGPT就是我的結對編程同伴,而且這個同伴很有耐心,有求必應,沒有偏見(?)。
在測試範圍和效率都是AI的優勢,能夠大幅度縮短撰寫測試代碼的時間,但在不洩漏敏感資訊的情況下,如何表達複雜的測試邏輯會是一個可能的問題。

完整流程

以下是透過提示ChatGPT來產生測試代碼,再讓開發者來撰寫產品代碼的過程。
首先告訴ChatGPT系統簡述和技術架構和需求,包含測試庫的要求(MSTest)。
目前的系統是一個溫度轉換程式。
技術架構:C#,Winform,用於模型的測試庫(MSTest)。
所有的程式碼都應該使用上述的技術架構來撰寫。
需求:程式應該提供華氏和攝氏的溫度轉換並以測試驅動開發來撰寫。
請你只提供單元測試,我再回答是否通過你的單元測試,以這種循環方式來完成整個過程

好的,以下是使用 MSTest 撰寫的兩個單元測試。一個用於驗證華氏到攝氏的溫度轉換,另一個用於驗證攝氏到華氏的溫度轉換。
using Microsoft.VisualStudio.TestTools.UnitTesting;
using YourTemperatureConversionNamespace; // 請將此替換為您的溫度轉換類別的命名空間

namespace TemperatureConversionTests
{
[TestClass]
public class TemperatureConversionTest
{
[TestMethod]
public void TestFahrenheitToCelsius()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputFahrenheit = 32.0;
double expectedCelsius = 0.0;

// Act
double actualCelsius = converter.FahrenheitToCelsius(inputFahrenheit);

// Assert
Assert.AreEqual(expectedCelsius, actualCelsius, 0.01);
}

[TestMethod]
public void TestCelsiusToFahrenheit()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputCelsius = 0.0;
double expectedFahrenheit = 32.0;

// Act
double actualFahrenheit = converter.CelsiusToFahrenheit(inputCelsius);

// Assert
Assert.AreEqual(expectedFahrenheit, actualFahrenheit, 0.01);
}
}
}
這裡假設你有一個 TemperatureConverter 類別,並且該類別有兩個方法:FahrenheitToCelsius 和 CelsiusToFahrenheit。這些方法都接受一個雙精度浮點數(double)並返回一個雙精度浮點數。這些方法應該進行溫度轉換。
請注意,這些測試中的精確度為 0.01,這意味著如果實際結果與預期結果之間的差異小於或等於 0.01,則該測試將被認為是通過的。
請將此單元測試代碼添加到你的測試項目中,並運行這兩個測試。如果你的溫度轉換代碼正確,則這兩個測試都應該通過。
接著就可以建立Winform專案,並在該專案中新增一個TemperatureConversionTest測試類別並直接貼上ChatGPT提供的完整代碼。
當然,執行測試一定會失敗,因為根本還沒有TemperatureConverter類別。
(如果對測試驅動開發有點了解,應該可以發現這也是驅動測試開發的流程,甚至也可以把ChatGPT當作結對編程的對象)
最後再根據ChatGPT提供的單元測試,實作被測代碼。為了避免洩漏敏感訊息,被測代碼不會提供給ChatGPT。
被測代碼如下
public class TemperatureConverter
{
public double FahrenheitToCelsius(double fahrenheit)
{
if (fahrenheit < -459.67)
{
throw new ArgumentOutOfRangeException("The Fahrenheit temperature cannot be less than -459.67");
}
return (fahrenheit - 32) * 5 / 9;
}

public double CelsiusToFahrenheit(double celsius)
{
if (celsius < -273.15)
{
throw new ArgumentOutOfRangeException("The Celsius temperature cannot be less than -273.15");
}
return (celsius * 9 / 5) + 32;
}
}
接著回答ChatGPT,已可通過單元測試,請再提供更多的單元測試。而且在這裡明確指定測試範圍為極限值
我已可通過你提供的單元測試,請再提供更多的單元測試,主要測試範圍為極限值
---
using Microsoft.VisualStudio.TestTools.UnitTesting;
using YourTemperatureConversionNamespace; // 請將此替換為您的溫度轉換類別的命名空間

namespace TemperatureConversionTests
{
[TestClass]
public class TemperatureConversionTest
{
[TestMethod]
public void TestFahrenheitToCelsiusAbsoluteZero()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputFahrenheit = -459.67;
double expectedCelsius = -273.15;

// Act
double actualCelsius = converter.FahrenheitToCelsius(inputFahrenheit);

// Assert
Assert.AreEqual(expectedCelsius, actualCelsius, 0.01);
}

[TestMethod]
public void TestCelsiusToFahrenheitAbsoluteZero()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputCelsius = -273.15;
double expectedFahrenheit = -459.67;

// Act
double actualFahrenheit = converter.CelsiusToFahrenheit(inputCelsius);

// Assert
Assert.AreEqual(expectedFahrenheit, actualFahrenheit, 0.01);
}

[TestMethod]
public void TestFahrenheitToCelsiusHighTemperature()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputFahrenheit = 212.0;
double expectedCelsius = 100.0;

// Act
double actualCelsius = converter.FahrenheitToCelsius(inputFahrenheit);

// Assert
Assert.AreEqual(expectedCelsius, actualCelsius, 0.01);
}

[TestMethod]
public void TestCelsiusToFahrenheitHighTemperature()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputCelsius = 100.0;
double expectedFahrenheit = 212.0;

// Act
double actualFahrenheit = converter.CelsiusToFahrenheit(inputCelsius);

// Assert
Assert.AreEqual(expectedFahrenheit, actualFahrenheit, 0.01);
}
}
}
把上述的單元測試新增到TemperatureConversionTest類別,並執行所有的單元測試,被測代碼可以成功通過單元測試!
接下來想對輸入的不合法數值做一些處理,所以要求ChatGPT對溫度低於絕對零度撰寫測試代碼
我已經可以通過你提供的單元測試,但如果輸入的溫度低於絕對零度,我想丟出一個異常,你可以寫出這種測試嗎?

當然可以。在這裡,我將為溫度低於絕對零度的情況寫出兩個測試。我們期望方法在這種情況下拋出一個異常。
using Microsoft.VisualStudio.TestTools.UnitTesting;
using System;
using YourTemperatureConversionNamespace; // 請將此替換為您的溫度轉換類別的命名空間

namespace TemperatureConversionTests
{
[TestClass]
public class TemperatureConversionTest
{
[TestMethod]
[ExpectedException(typeof(ArgumentOutOfRangeException))]
public void TestFahrenheitToCelsiusBelowAbsoluteZero()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputFahrenheit = -500.0; // Less than absolute zero

// Act
converter.FahrenheitToCelsius(inputFahrenheit);

// Assert is handled by the ExpectedException attribute
}

[TestMethod]
[ExpectedException(typeof(ArgumentOutOfRangeException))]
public void TestCelsiusToFahrenheitBelowAbsoluteZero()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputCelsius = -300.0; // Less than absolute zero

// Act
converter.CelsiusToFahrenheit(inputCelsius);

// Assert is handled by the ExpectedException attribute
}
}
}
在這兩個測試中,我們使用了 ExpectedException 屬性來指定我們期望方法在給定的情況下拋出的異常類型。在這種情況下,我們期望方法在溫度低於絕對零度時拋出 ArgumentOutOfRangeException。如果你的溫度轉換方法正確地處理了這種情況,那麼這些測試應該會通過。
把ChatGPT新增的測試代碼貼到測試類別之後,執行單元測試應該會失敗。因為在TemperatureConverter沒有根據輸入的不合法數據丟出異常。
回頭修改TemperatureConverter,主要針對華氏低於-459.67和攝氏低於-243.15,丟出異常
public class TemperatureConverter
{
public double FahrenheitToCelsius(double fahrenheit)
{
if (fahrenheit < -459.67)
{
throw new ArgumentOutOfRangeException("The Fahrenheit temperature cannot be less than -459.67");
}
return (fahrenheit - 32) * 5 / 9;
}

public double CelsiusToFahrenheit(double celsius)
{
if (celsius < -273.15)
{
throw new ArgumentOutOfRangeException("The Celsius temperature cannot be less than -273.15");
}
return (celsius * 9 / 5) + 32;
}
}
完成後順利通過單元測試!
最後想請ChatGPT提供它覺得缺乏的單元測試。
我已經可以通過單元測試了,最後我想請提供你覺得缺乏的單元測試

很好,你已經涵蓋了一些重要的測試案例。除了我們已經討論過的測試案例,我們還可以添加一些邊界測試和特殊值測試。以下是一些我們可以添加的額外測試:
  1. 輸入為零的情況:零度華氏對應的攝氏溫度為 -17.78°C,零度攝氏對應的華氏溫度為 32°F。
  2. 一些隨機值的測試:選擇一些隨機的華氏和攝氏溫度值,檢查轉換是否正確。
以下是相應的單元測試:
using Microsoft.VisualStudio.TestTools.UnitTesting;
using YourTemperatureConversionNamespace; // 請將此替換為您的溫度轉換類別的命名空間

namespace TemperatureConversionTests
{
[TestClass]
public class TemperatureConversionTest
{
[TestMethod]
public void TestFahrenheitToCelsiusZero()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputFahrenheit = 0.0;
double expectedCelsius = -17.78;

// Act
double actualCelsius = converter.FahrenheitToCelsius(inputFahrenheit);

// Assert
Assert.AreEqual(expectedCelsius, actualCelsius, 0.01);
}

[TestMethod]
public void TestCelsiusToFahrenheitZero()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputCelsius = 0.0;
double expectedFahrenheit = 32.0;

// Act
double actualFahrenheit = converter.CelsiusToFahrenheit(inputCelsius);

// Assert
Assert.AreEqual(expectedFahrenheit, actualFahrenheit, 0.01);
}

[TestMethod]
public void TestFahrenheitToCelsiusRandom()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputFahrenheit = 45.0; // A random Fahrenheit value
double expectedCelsius = 7.22; // Corresponding Celsius value

// Act
double actualCelsius = converter.FahrenheitToCelsius(inputFahrenheit);

// Assert
Assert.AreEqual(expectedCelsius, actualCelsius, 0.01);
}

[TestMethod]
public void TestCelsiusToFahrenheitRandom()
{
// Arrange
TemperatureConverter converter = new TemperatureConverter();
double inputCelsius = 20.0; // A random Celsius value
double expectedFahrenheit = 68.0; // Corresponding Fahrenheit value

// Act
double actualFahrenheit = converter.CelsiusToFahrenheit(inputCelsius);

// Assert
Assert.AreEqual(expectedFahrenheit, actualFahrenheit, 0.01);
}
}
}
這些測試將增加對你的溫度轉換程式碼的信心。如果所有這些測試都通過了,那麼你的溫度轉換程式碼在這些測試覆蓋的範圍內應該是正確的。請注意,單元測試能幫助我們提高對程式碼的信心,但並不能保證程式碼百分之百的正確。例如,有可能存在我們尚未考慮到的邊界情況或特殊情況。因此,雖然這些測試是一個很好的起點,但可能仍需要進一步的測試和審查來確保程式碼的健壯性。
avatar-img
1會員
6內容數
目前專注於軟體設計架構領域,旨在分享並提供良好且合適的原則 / 模式 / 系統 / 開發法。
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
寫狐的沙龍 的其他內容
前同事阿強是開發部門的技術強者,最近因為大功一件加上考績滿分,被老闆提拔為經理。 看起來阿強走路有風。沒想到在聚會時卻跑來跟我訴苦。 “唉我喜歡技術也專精技術,可是我從沒管過人,也沒想過要管人” “你有跟你老闆談過你沒意願嗎?” “好吧…(難道技術能力強也代表管理能力強嗎)” 答案是不一定!
甚麼是技術雷達? 技術雷達是一種評估工具,用來判斷軟體領域中各種技術並對這些技術應採取什麼行動的方法。(來源為 ThoughtWorks,參閱這裡) 象限 技術雷達本身是一個切割成4等份的圓形,這4等份各為技巧(左上),工具(左下),語言框架(右上),平台(右下),代表技術種類(在官方網站中稱為象限
無論是參與面試或釐清需求,系統設計常常令人感到困惑,不知如何進行。 我想,原因大概有兩種。 首先,不同領域的系統通常都有不同的設計要素,如何找出這些要素進行提問? 其次,即便提出疑問找出重點,接下來要如何針對這些要素進行高層設計或規劃? 《系統設計面試指南》對於系統設計的初步規劃或高層結構提供合適的
若說大多數成功人士共通的要素是什麼,我想好習慣絕對是其中之一。好習慣很重要,但要保持好習慣也很困難。 很多人以短暫的衝勁,熱情來激勵自我。常見的說法就是『今年要減肥』,『這禮拜要戒菸』。 另一些人會製作計劃表,期許透過嚴謹且完美的計畫,達成目標。 可惜,熱情抵不過人性,計劃趕不上變化。 好習慣的養成
當我們胸懷大志,勇往直前時,最大的阻礙往往來自本身或他人的非理性思考。這些非理性思考常讓人自我懷疑,中途放棄。 要如何免於自我設限的思維?  大局思維透過行為科學揪出人們的非理性思考並透過一步步設定具體目標和規律步驟達成自我理想的實現,翻轉人生。
前同事阿強是開發部門的技術強者,最近因為大功一件加上考績滿分,被老闆提拔為經理。 看起來阿強走路有風。沒想到在聚會時卻跑來跟我訴苦。 “唉我喜歡技術也專精技術,可是我從沒管過人,也沒想過要管人” “你有跟你老闆談過你沒意願嗎?” “好吧…(難道技術能力強也代表管理能力強嗎)” 答案是不一定!
甚麼是技術雷達? 技術雷達是一種評估工具,用來判斷軟體領域中各種技術並對這些技術應採取什麼行動的方法。(來源為 ThoughtWorks,參閱這裡) 象限 技術雷達本身是一個切割成4等份的圓形,這4等份各為技巧(左上),工具(左下),語言框架(右上),平台(右下),代表技術種類(在官方網站中稱為象限
無論是參與面試或釐清需求,系統設計常常令人感到困惑,不知如何進行。 我想,原因大概有兩種。 首先,不同領域的系統通常都有不同的設計要素,如何找出這些要素進行提問? 其次,即便提出疑問找出重點,接下來要如何針對這些要素進行高層設計或規劃? 《系統設計面試指南》對於系統設計的初步規劃或高層結構提供合適的
若說大多數成功人士共通的要素是什麼,我想好習慣絕對是其中之一。好習慣很重要,但要保持好習慣也很困難。 很多人以短暫的衝勁,熱情來激勵自我。常見的說法就是『今年要減肥』,『這禮拜要戒菸』。 另一些人會製作計劃表,期許透過嚴謹且完美的計畫,達成目標。 可惜,熱情抵不過人性,計劃趕不上變化。 好習慣的養成
當我們胸懷大志,勇往直前時,最大的阻礙往往來自本身或他人的非理性思考。這些非理性思考常讓人自我懷疑,中途放棄。 要如何免於自我設限的思維?  大局思維透過行為科學揪出人們的非理性思考並透過一步步設定具體目標和規律步驟達成自我理想的實現,翻轉人生。
你可能也想看
Google News 追蹤
Thumbnail
我是一位提示工程師 生成式AI的課程,我大概已經上過1,000人次以上,最近更花費重金,在台大霖澤館舉辦一場百人AIBOT設計課程,並在將近10位的助教協助下,完成課程的設定目標。 在生成式AI的領域中,變化實在是非常地迅速,今天表現不佳的ChatGPT,明天可能問題就可以解決。 過去還有想辦
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 再來談一個使用 TOCD 這個 Prompt 範本,建立「食譜推薦」的完整 Prompt,TOCD 的說
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 延續 AI說書 - Prompt Engineering - 40 ,目標是要使用 TOCD 這個 Pr
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 在複雜的軟體開發和人工智慧世界中,設計模板作為概念工具出現,幫助工程師巧妙地應對複雜的情況,當涉及 Pr
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 ChatGPT縱然成功,但是仍有一些任務處理得不好,例如: 需要邏輯推論的任務 多步驟的任務
Thumbnail
本文談論了使用ChatGPT的心得和學習新技術的10個步驟,透過這些方法和步驟可以幫助學習者快速有效地掌握新技術。
Thumbnail
給出好的指令也是一門學問! 我們在業界稱為「指令工程」(Prompt Engineering),OpenAI 官方也有公布指令工程的操作資訊,但是今天我們只要學會這樣的操作方式,你可以贏過現在多數的AI使用者。
Thumbnail
OpenAI推出的Custom GPTs可以讓你自己量身製作符合你自己需求的AI助手,客製化AI工具喂給他不同的資料產生的結果好壞程度也差很多,好的AI工具能夠幫大家更有效率解決很多問題,不過因為製作門檻很低,有非常大量的客製化GPTs已經在OpenAI的ChatGPT上,這篇推薦給大家的是蒐集整理
Thumbnail
從最初的接觸到深度的投懷送抱,這篇文章對於解決如何安裝text-generation-webui提供了清晰的指引。 它引導讀者通過它的官方方式和Docker方式做到這一點。此外,作者還分享了必要的優化建議。
Thumbnail
本文章介紹瞭如何使用AutoGPT, 一種可以自主蒐集資料生成訴求,幫你與ChatGPT在互動中提出一連串的問題,來解決你的問題。對於安裝時的常見問題也進行了解答,並提供了使用的步驟以及目前的解決方式。
Thumbnail
我是一位提示工程師 生成式AI的課程,我大概已經上過1,000人次以上,最近更花費重金,在台大霖澤館舉辦一場百人AIBOT設計課程,並在將近10位的助教協助下,完成課程的設定目標。 在生成式AI的領域中,變化實在是非常地迅速,今天表現不佳的ChatGPT,明天可能問題就可以解決。 過去還有想辦
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 再來談一個使用 TOCD 這個 Prompt 範本,建立「食譜推薦」的完整 Prompt,TOCD 的說
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 延續 AI說書 - Prompt Engineering - 40 ,目標是要使用 TOCD 這個 Pr
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 在複雜的軟體開發和人工智慧世界中,設計模板作為概念工具出現,幫助工程師巧妙地應對複雜的情況,當涉及 Pr
我們人類和ChatGPT的對話技巧也是需要學習的,有鑑於此,我想要一天分享一點「和ChatGPT對話的技術」,並且每篇文章長度控制在三分鐘以內,讓大家不會壓力太大,但是又能夠每天成長一點。 ChatGPT縱然成功,但是仍有一些任務處理得不好,例如: 需要邏輯推論的任務 多步驟的任務
Thumbnail
本文談論了使用ChatGPT的心得和學習新技術的10個步驟,透過這些方法和步驟可以幫助學習者快速有效地掌握新技術。
Thumbnail
給出好的指令也是一門學問! 我們在業界稱為「指令工程」(Prompt Engineering),OpenAI 官方也有公布指令工程的操作資訊,但是今天我們只要學會這樣的操作方式,你可以贏過現在多數的AI使用者。
Thumbnail
OpenAI推出的Custom GPTs可以讓你自己量身製作符合你自己需求的AI助手,客製化AI工具喂給他不同的資料產生的結果好壞程度也差很多,好的AI工具能夠幫大家更有效率解決很多問題,不過因為製作門檻很低,有非常大量的客製化GPTs已經在OpenAI的ChatGPT上,這篇推薦給大家的是蒐集整理
Thumbnail
從最初的接觸到深度的投懷送抱,這篇文章對於解決如何安裝text-generation-webui提供了清晰的指引。 它引導讀者通過它的官方方式和Docker方式做到這一點。此外,作者還分享了必要的優化建議。
Thumbnail
本文章介紹瞭如何使用AutoGPT, 一種可以自主蒐集資料生成訴求,幫你與ChatGPT在互動中提出一連串的問題,來解決你的問題。對於安裝時的常見問題也進行了解答,並提供了使用的步驟以及目前的解決方式。