<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:content="http://purl.org/rss/1.0/modules/content/"><channel><title>Representativeness on Tarragon</title><link>https://tarrragon.github.io/blog/tags/representativeness/</link><description>Recent content in Representativeness on Tarragon</description><generator>Hugo -- gohugo.io</generator><language>zh-TW</language><copyright>Tarragon (CC BY 4.0)</copyright><lastBuildDate>Fri, 19 Jun 2026 00:00:00 +0000</lastBuildDate><atom:link href="https://tarrragon.github.io/blog/tags/representativeness/index.xml" rel="self" type="application/rss+xml"/><item><title>Test data 代表性</title><link>https://tarrragon.github.io/blog/testing/05-test-design-judgment/test-data-representativeness/</link><pubDate>Fri, 19 Jun 2026 00:00:00 +0000</pubDate><guid>https://tarrragon.github.io/blog/testing/05-test-design-judgment/test-data-representativeness/</guid><description>&lt;p>測試資料的代表性是指測試輸入能多大程度反映真實環境的輸入分佈。「測試資料能代表真實環境」是每個 test 的隱性假設 — 這個假設成立時 test 有效，不成立時 test 通過但問題仍在。&lt;/p>
&lt;h2 id="代表性問題的案例">代表性問題的案例&lt;/h2>
&lt;p>app_tunnel 的 ANSI parser 有 18 個 test，全部通過。測試資料是手寫的 SGR 色彩碼（&lt;code>\x1B[31mhello\x1B[0m&lt;/code>），parser 正確解析這類序列。&lt;/p>
&lt;p>真實 zsh 啟動後送出的控制序列包含 OSC 標題設定、CSI private mode、字元集指定等至少 5 種類型。Parser 只認識 SGR，其他全部透傳為亂碼（&lt;a href="https://tarrragon.github.io/blog/testing/cases/ansi-parser-test-data-blindspot/" data-link-title="T.C3 ANSI parser 測試資料不覆蓋真實 shell output" data-link-desc="ANSI parser 只處理基本 SGR 色彩碼、unit test 用手寫乾淨字串驗證 — 真實 zsh prompt 送出 OSC 標題設定、CSI private mode 游標隱藏、括號貼上模式等數十種控制序列，全部殘留為亂碼">T.C3&lt;/a>）。&lt;/p>
&lt;p>18 個 test 覆蓋了 1 種序列類型。測試資料的代表性假設（「SGR 就是主要的序列類型」）和真實環境不符。&lt;/p>
&lt;h2 id="三種測試資料來源">三種測試資料來源&lt;/h2>
&lt;h3 id="手寫">手寫&lt;/h3>
&lt;p>開發者根據對輸入格式的理解手動建構測試字串。&lt;/p>
&lt;p>優點：精確控制、容易理解、可以針對特定邊界條件設計。&lt;/p>
&lt;p>缺點：受限於開發者對輸入分佈的認知。如果開發者不知道真實環境有哪些輸入類型，手寫的測試資料就是開發者認知的子集 — T.C3 就是這個模式。&lt;/p>
&lt;p>適合場景：格式規格明確且有限（JSON schema、固定格式的設定檔）、邊界條件測試（空值、最大長度、特殊字元）。&lt;/p>
&lt;h3 id="錄製">錄製&lt;/h3>
&lt;p>從真實環境擷取實際的輸入資料，作為 test 的輸入。&lt;/p>
&lt;p>優點：直接反映真實環境的輸入分佈，包含開發者不知道的輸入類型。&lt;/p>
&lt;p>缺點：錄製的資料可能包含敏感資訊（需要脫敏）、資料量可能大（需要挑選代表性樣本）、真實環境的輸入可能隨時間改變（錄製的資料可能過時）。&lt;/p>
&lt;p>適合場景：輸入格式複雜且規格不完整（終端機 escape 序列、網路封包、使用者產生的內容）、parser 類的功能（需要知道「真實輸入長什麼樣」）。&lt;/p>
&lt;p>T.C3 如果用錄製的真實 zsh 啟動輸出作為測試資料，OSC 和 CSI private mode 會自然出現在輸入中。即使 parser 仍然不處理這些序列，test 至少能讓開發者看到「有 5 種序列類型，我只處理了 1 種」。&lt;/p>
&lt;h3 id="生成property-based-testing">生成（Property-based testing）&lt;/h3>
&lt;p>用 generator 自動產生大量隨機或半隨機的輸入，驗證 parser 的行為是否符合通用性質（不崩潰、輸出長度 &amp;lt;= 輸入長度、冪等性）。&lt;/p>
&lt;p>優點：覆蓋人類想不到的 edge case、發現意外的崩潰或無限迴圈。&lt;/p>
&lt;p>缺點：不針對特定功能驗證（驗證的是通用性質，不是「OSC 序列是否被正確處理」）、generator 本身需要維護。&lt;/p>
&lt;p>適合場景：parser、serializer、codec 等輸入格式複雜的功能。和手寫 test 互補 — 手寫驗證特定行為正確性，生成驗證通用穩定性。&lt;/p>
&lt;h2 id="兩類-test-的分工">兩類 test 的分工&lt;/h2>
&lt;p>T.C3 的策略建議是把 test 分成兩類：&lt;/p>
&lt;p>&lt;strong>功能正確性 test&lt;/strong>：用手寫乾淨字串驗證 parser 對已知序列的處理正確性。&lt;code>\x1B[31mhello\x1B[0m&lt;/code> 應該產生紅色 token — 這是功能規格的驗證。&lt;/p>
&lt;p>&lt;strong>環境相容性 test&lt;/strong>：用錄製的真實輸出驗證 parser 在真實環境中的表現。不斷言「每個序列都被正確處理」，而是斷言「沒有崩潰」「沒有未處理序列殘留在可見輸出中」。&lt;/p>
&lt;p>兩類 test 回答不同問題。功能正確性回答「parser 的邏輯對不對」，環境相容性回答「parser 在真實環境中夠不夠用」。&lt;/p>
&lt;h2 id="下一步路由">下一步路由&lt;/h2>
&lt;ul>
&lt;li>Assertion 的品質判斷 → &lt;a href="https://tarrragon.github.io/blog/testing/05-test-design-judgment/assertion-quality/" data-link-title="Assertion 品質三問" data-link-desc="斷言的是行為嗎？能區分正確和錯誤嗎？會 flaky 嗎？— 三個問題判斷 assertion 是否有效">Assertion 品質三問&lt;/a>&lt;/li>
&lt;li>Mock 邊界的判斷 → &lt;a href="https://tarrragon.github.io/blog/testing/05-test-design-judgment/mock-boundary-decision/" data-link-title="Mock 邊界判斷決策表" data-link-desc="什麼時候 mock 夠用、什麼時候需要真實服務 — 從 API 層 / 協議層 / 環境層的斷裂點判斷 mock 的適用範圍">Mock 邊界判斷決策表&lt;/a>&lt;/li>
&lt;li>Protocol integration test 用真實服務輸出 → &lt;a href="https://tarrragon.github.io/blog/testing/03-protocol-integration-test/websocket-protocol-test/" data-link-title="WebSocket 協議測試實作" data-link-desc="對真實 ttyd 驗證 frame type 和 auth handshake — 從 T.C1 和 T.C2 的教訓推導出的 protocol integration test 設計">testing 模組三 WebSocket 協議測試&lt;/a>&lt;/li>
&lt;/ul></description><content:encoded><![CDATA[<p>測試資料的代表性是指測試輸入能多大程度反映真實環境的輸入分佈。「測試資料能代表真實環境」是每個 test 的隱性假設 — 這個假設成立時 test 有效，不成立時 test 通過但問題仍在。</p>
<h2 id="代表性問題的案例">代表性問題的案例</h2>
<p>app_tunnel 的 ANSI parser 有 18 個 test，全部通過。測試資料是手寫的 SGR 色彩碼（<code>\x1B[31mhello\x1B[0m</code>），parser 正確解析這類序列。</p>
<p>真實 zsh 啟動後送出的控制序列包含 OSC 標題設定、CSI private mode、字元集指定等至少 5 種類型。Parser 只認識 SGR，其他全部透傳為亂碼（<a href="/blog/testing/cases/ansi-parser-test-data-blindspot/" data-link-title="T.C3 ANSI parser 測試資料不覆蓋真實 shell output" data-link-desc="ANSI parser 只處理基本 SGR 色彩碼、unit test 用手寫乾淨字串驗證 — 真實 zsh prompt 送出 OSC 標題設定、CSI private mode 游標隱藏、括號貼上模式等數十種控制序列，全部殘留為亂碼">T.C3</a>）。</p>
<p>18 個 test 覆蓋了 1 種序列類型。測試資料的代表性假設（「SGR 就是主要的序列類型」）和真實環境不符。</p>
<h2 id="三種測試資料來源">三種測試資料來源</h2>
<h3 id="手寫">手寫</h3>
<p>開發者根據對輸入格式的理解手動建構測試字串。</p>
<p>優點：精確控制、容易理解、可以針對特定邊界條件設計。</p>
<p>缺點：受限於開發者對輸入分佈的認知。如果開發者不知道真實環境有哪些輸入類型，手寫的測試資料就是開發者認知的子集 — T.C3 就是這個模式。</p>
<p>適合場景：格式規格明確且有限（JSON schema、固定格式的設定檔）、邊界條件測試（空值、最大長度、特殊字元）。</p>
<h3 id="錄製">錄製</h3>
<p>從真實環境擷取實際的輸入資料，作為 test 的輸入。</p>
<p>優點：直接反映真實環境的輸入分佈，包含開發者不知道的輸入類型。</p>
<p>缺點：錄製的資料可能包含敏感資訊（需要脫敏）、資料量可能大（需要挑選代表性樣本）、真實環境的輸入可能隨時間改變（錄製的資料可能過時）。</p>
<p>適合場景：輸入格式複雜且規格不完整（終端機 escape 序列、網路封包、使用者產生的內容）、parser 類的功能（需要知道「真實輸入長什麼樣」）。</p>
<p>T.C3 如果用錄製的真實 zsh 啟動輸出作為測試資料，OSC 和 CSI private mode 會自然出現在輸入中。即使 parser 仍然不處理這些序列，test 至少能讓開發者看到「有 5 種序列類型，我只處理了 1 種」。</p>
<h3 id="生成property-based-testing">生成（Property-based testing）</h3>
<p>用 generator 自動產生大量隨機或半隨機的輸入，驗證 parser 的行為是否符合通用性質（不崩潰、輸出長度 &lt;= 輸入長度、冪等性）。</p>
<p>優點：覆蓋人類想不到的 edge case、發現意外的崩潰或無限迴圈。</p>
<p>缺點：不針對特定功能驗證（驗證的是通用性質，不是「OSC 序列是否被正確處理」）、generator 本身需要維護。</p>
<p>適合場景：parser、serializer、codec 等輸入格式複雜的功能。和手寫 test 互補 — 手寫驗證特定行為正確性，生成驗證通用穩定性。</p>
<h2 id="兩類-test-的分工">兩類 test 的分工</h2>
<p>T.C3 的策略建議是把 test 分成兩類：</p>
<p><strong>功能正確性 test</strong>：用手寫乾淨字串驗證 parser 對已知序列的處理正確性。<code>\x1B[31mhello\x1B[0m</code> 應該產生紅色 token — 這是功能規格的驗證。</p>
<p><strong>環境相容性 test</strong>：用錄製的真實輸出驗證 parser 在真實環境中的表現。不斷言「每個序列都被正確處理」，而是斷言「沒有崩潰」「沒有未處理序列殘留在可見輸出中」。</p>
<p>兩類 test 回答不同問題。功能正確性回答「parser 的邏輯對不對」，環境相容性回答「parser 在真實環境中夠不夠用」。</p>
<h2 id="下一步路由">下一步路由</h2>
<ul>
<li>Assertion 的品質判斷 → <a href="/blog/testing/05-test-design-judgment/assertion-quality/" data-link-title="Assertion 品質三問" data-link-desc="斷言的是行為嗎？能區分正確和錯誤嗎？會 flaky 嗎？— 三個問題判斷 assertion 是否有效">Assertion 品質三問</a></li>
<li>Mock 邊界的判斷 → <a href="/blog/testing/05-test-design-judgment/mock-boundary-decision/" data-link-title="Mock 邊界判斷決策表" data-link-desc="什麼時候 mock 夠用、什麼時候需要真實服務 — 從 API 層 / 協議層 / 環境層的斷裂點判斷 mock 的適用範圍">Mock 邊界判斷決策表</a></li>
<li>Protocol integration test 用真實服務輸出 → <a href="/blog/testing/03-protocol-integration-test/websocket-protocol-test/" data-link-title="WebSocket 協議測試實作" data-link-desc="對真實 ttyd 驗證 frame type 和 auth handshake — 從 T.C1 和 T.C2 的教訓推導出的 protocol integration test 設計">testing 模組三 WebSocket 協議測試</a></li>
</ul>
]]></content:encoded></item></channel></rss>