AI辅助编程的边界探索:当Copilot学会写测试

AI辅助编程的边界探索:当Copilot学会写测试
在这里插入图片描述
👋 大家好,欢迎来到我的技术博客!
📚 在这里,我会分享学习笔记、实战经验与技术思考,力求用简单的方式讲清楚复杂的问题。
🎯 本文将围绕人工智能这个话题展开,希望能为你带来一些启发或实用的参考。
🌱 无论你是刚入门的新手,还是正在进阶的开发者,希望你都能有所收获!

文章目录

AI辅助编程的边界探索:当Copilot学会写测试 🚀

在过去的几年里,我们见证了人工智能辅助编程工具(如 GitHub Copilot、Cursor 等)从简单的代码补全插件,进化成为能够独立思考、生成复杂逻辑的“数字伙伴”。大多数开发者的使用场景停留在“帮我写个排序算法”或者“帮我补全这个React组件的样式”。然而,当AI开始染指软件工程中最繁琐、最需要逻辑严密性的领域——测试(Testing) 时,一切都变得有趣且充满挑战。

今天,我们就来深入探讨一下:当Copilot学会了写测试,它的边界在哪里?它能否取代人工编写测试用例的工作?它又会在哪里“翻车”? 让我们通过一系列真实的代码实验来寻找答案 📝。


1. 从“写代码”到“验代码”:AI的新战场 ⚔️

传统的编程教学往往强调如何实现功能(Implementation),而软件工程的核心却有一半是维护。维护的核心不是加功能,而是保证加功能不破坏旧功能。这就使得测试变得至关重要。

对于AI来说,写一段计算斐波那契数列的代码是简单的,因为逻辑是收敛的。但是,写一段针对斐波那契数列的测试用例则是发散的:你要考虑输入0、输入负数、输入超大数、溢出处理等等。AI写测试的核心难点在于:它不知道产品的“意图”(Intent),它只能根据上下文进行“模仿”。

下面,我们构建一个典型的业务场景,来测试Copilot的能力边界。我们将使用 TypeScriptJest 作为实验环境。

场景设定:一个简单的支付网关模拟器 💳

我们需要一个看似简单但包含逻辑分支的函数。

// paymentService.tsinterfacePaymentRequest{ userId:string; amount:number; currency:'USD'|'CNY'|'EUR';}interfacePaymentResult{ success:boolean; transactionId?:string; error?:string;}exportconst processPayment =(req: PaymentRequest): PaymentResult =>{// 业务逻辑:金额必须大于0if(req.amount <=0){return{ success:false, error:'Invalid amount'};}// 业务逻辑:如果是测试用户,直接返回成功if(req.userId.startsWith('test_')){return{ success:true, transactionId:'test_tx_123'};}// 模拟外部支付API调用(这里假设总是成功)const txId =`tx_${Math.random().toString(36).substr(2,9)}`;return{ success:true, transactionId: txId };};

现在,如果我们直接告诉AI:“为这个文件写测试”,会发生什么?


2. 初级实验:AI能写出“Happy Path”吗? ✅

“Happy Path”(快乐路径)指的是测试用例中最理想、最常见的流程,比如“用户付了钱,系统返回成功”。

Prompt:// Write unit tests for processPayment function using Jest

Copilot (or similar AI) 生成的代码:

// paymentService.test.tsimport{ processPayment }from'./paymentService';describe('processPayment',()=>{// 测试正常的正向流程it('should return success for valid payment',()=>{const request ={ userId:'user_123', amount:100, currency:'USD',};const result =processPayment(request);expect(result.success).toBe(true);expect(result.transactionId).toBeDefined();});// 测试金额为0的情况it('should fail when amount is zero',()=>{const request ={ userId:'user_123', amount:0, currency:'USD',};const result =processPayment(request);expect(result.success).toBe(false);expect(result.error).toBe('Invalid amount');});});

分析: 这是标准的“模板式”测试。AI完美地识别了代码中的两个 if 分支,并生成了对应的断言。这是 Copilot 最擅长的地方,它本质上是在做“模式匹配”。

但是,这引出了我们的第一个疑问:如果我们不告诉它需要测试“负数金额”,它会主动写吗?通常不会。这暴露了AI的第一个盲区:被动响应,而非主动发现缺陷。


3. 进阶实验:Mocking 与 外部依赖 🎭

在真实的项目中,你的函数通常不是独立的,它会调用数据库、调用第三方API(Stripe, PayPal等)。测试必须将这些外部依赖 Mock(模拟) 掉。这才是考验AI逻辑理解能力的地方。

让我们修改代码,引入一个真实的异步外部调用。

// database.ts (模拟数据库)exportconstsaveTransaction=async(data:any)=>{// 这里连接了真实的数据库return{ id:'db_123',...data };};
// orderService.tsimport{ saveTransaction }from'./database';exportconstcreateOrder=async(items:string[], userId:string)=>{if(!items.length)thrownewError('Cart is empty');const order ={ items, userId, createdAt:newDate()};const saved =awaitsaveTransaction(order);return saved;};

Prompt:// Write tests for createOrder and mock the database module

AI 生成的代码:

// orderService.test.tsimport{ createOrder }from'./orderService';import{ saveTransaction }from'./database';// 关键步骤:Mocking jest.mock('./database',()=>({ saveTransaction: jest.fn(),}));describe('createOrder',()=>{it('should save order to database',async()=>{// Arrangeconst mockSave = saveTransaction as jest.Mock; mockSave.mockResolvedValue({ id:'mock_id'});// Actconst result =awaitcreateOrder(['item1'],'user_1');// Assertexpect(mockSave).toHaveBeenCalled();expect(result.id).toBe('mock_id');});it('should throw error if cart is empty',async()=>{awaitexpect(createOrder([],'user_1')).rejects.toThrow('Cart is empty');});});

** verdict (评判)😗*
这一波操作非常漂亮 😎。AI不仅自动生成了 jest.mock,还正确处理了异步函数 (async/await)。这说明在基于上下文的单元测试场景下,AI已经达到了相当高的可用性。

但是,精彩的地方来了。如果我们稍微修改一下业务逻辑,增加一个边界条件,AI的表现会如何?


4. 陷阱与幻觉:AI写测试时犯的那些错 🤪

AI并不真正“理解”代码,它只是根据统计学预测最可能的下一段文本。这导致它在处理边界值特定业务规则时,会产生令人啼笑皆非的“幻觉测试”。

案例 A:永远不会错的测试

假设我们有一段非常糟糕的代码(或者AI根本看不懂的混淆代码):

// weirdLogic.tsexportconstgetStatus=(code:number)=>{if(code >100)return'OK';return'ERROR';};

如果你让AI测试它,AI会根据代码生成:

it('should return OK for code > 100',()=>{expect(getStatus(101)).toBe('OK');});

这看起来没问题。但是,如果产品需求变了,实际上业务规则应该是 code >= 100(大于等于)呢?AI不会质疑代码,它会忠实且错误地测试错误的代码。这就是所谓的 “测试覆盖了代码,但验证了错误”

案例 B:永远跑不通的断言

有时候,AI会写出看似合理,但运行时永远无法通过的断言。

Prompt:// Test that the array contains unique elements

假设我们的原始数组是 [1, 2, 3, 3](有重复),AI可能会生成这样的测试:

it('should contain only unique elements',()=>{// 假设这里返回了 [1, 2, 3, 3]const result =getData();const unique =newSet(result);// AI的幻觉:它假设 Set 后长度等于原始长度expect(unique.size).toBe(result.length);});

如果你运行这个测试,它会 Fail。但开发者看到 Fail 后,往往会去改代码(试图让数组没有重复),而不是意识到这个测试用例本身是基于错误的假设(数据源本身可能有重复,这不是一个Bug,而是一个业务事实)。AI制造了新的“伪Bug”。

案例 C:复杂集成测试的无力

在端到端(E2E)测试或者集成测试中,AI的表现会指数级下降。因为这需要AI理解状态(State)

想象一下,你需要测试:“用户登录 -> 添加购物车 -> 点击结算 -> 验证库存扣减 -> 验证支付回调 -> 验证数据库订单状态”。

这种长达10步的流程,AI很难一次性生成完整的 Flow。除非你在极其细粒度的 Prompt 中,详细描述每一步的数据库状态和预期结果。


5. 人机协作:重新定义测试工作流 🤝

既然AI不是万能的,那么最佳的策略就是利用AI的吞吐量,配合人类的判断力。让我们用 Mermaid 图表来展示这个新时代的测试工作流。

在这个流程中,AI不再是执行者,而是协作者

渲染错误: Mermaid 渲染失败: Parse error on line 3: ...--> C[AI: 自动生成基础测试桩 (Stubs)] C --> D -----------------------^ Expecting 'SQE', 'DOUBLECIRCLEEND', 'PE', '-)', 'STADIUMEND', 'SUBROUTINEEND', 'PIPE', 'CYLINDEREND', 'DIAMOND_STOP', 'TAGEND', 'TRAPEND', 'INVTRAPEND', 'UNICODE_TEXT', 'TEXT', 'TAGSTART', got 'PS'

如上图所示,AI负责第一层的覆盖(Happy path + 基本的 Mock),而人类负责最深层的逻辑校验

实践技巧:如何高效地让AI写测试?

  1. 不要只说“写测试”
    • // write tests
    • // Write unit tests for this function, specifically testing the error handling when input is null and the currency conversion logic for USD to EUR.
  2. 指定测试框架
    明确告诉它用什么工具(例如 Jest, Mocha, PyTest),它能生成更精准的脚手架。
  3. 让它扮演 QA 工程师
    使用 Prompt: Act as a QA engineer. What edge cases should I test for this login function? (扮演QA工程师,我应该测试这个登录函数的哪些边界情况?)。通常它会列出一堆你没想到的negative case,比如SQL注入测试、并发登录测试等。

6. 展望未来:AI会取代测试工程师吗? 🔮

目前的AI,包括Copilot,在测试领域的定位应该是 “高级助理”,而不是“独立工作者”。

AI目前的强项:

  • 生成样板代码(Boilerplate)。
  • 编写针对确定逻辑的单元测试。
  • 批量生成 Mock 代码。
  • 将代码转化为测试(如果你有一段代码,它能帮你写测试)。

AI目前的弱项:

  • 理解真实的业务意图(Context)。
  • 编写性能测试和压力测试。
  • 维护测试(当代码重构后,测试需要同步更新,AI往往会照着旧代码生成新的错误测试)。
  • 探索性测试(Exploratory Testing)。

结论是:Copilot能把测试的门槛降低,但无法把测试的质量拉高。 质量取决于人类对业务的理解深度。

最后,推荐大家在使用AI辅助编程时,配合阅读一下官方的最佳实践,比如 Jest 官方文档 中关于异步测试的章节,或者 MDN Web Docs 上关于单元测试的基础概念。

让我们拥抱变化,利用工具,但永远保持怀疑精神 👀。毕竟,代码可能会说谎,但测试不能。


如果你想了解更多关于TDD(测试驱动开发)的哲学,可以参考 Wikipedia - Test-driven development


🙌 感谢你读到这里!
🔍 技术之路没有捷径,但每一次阅读、思考和实践,都在悄悄拉近你与目标的距离。
💡 如果本文对你有帮助,不妨 👍 点赞、📌 收藏、📤 分享 给更多需要的朋友!
💬 欢迎在评论区留下你的想法、疑问或建议,我会一一回复,我们一起交流、共同成长 🌿
🔔 关注我,不错过下一篇干货!我们下期再见!✨

Read more

AIGC浪潮下,图文内容社区数据指标体系如何构建?

AIGC浪潮下,图文内容社区数据指标体系如何构建?

文章目录 * 01 案例:以图文内容社区为例实践数据指标体构建 * 02 4个步骤实现数据指标体系构建 * 1. 明确业务目标,梳理北极星指标 * 2. 梳理业务流程,明确过程指标 * 3. 指标下钻分级,构建多层级数据指标体系 * 4. 添加分析维度,构建完整的数据指标体系 * 03 构建数据指标体系的过程总结 * 作者简介 * 目 录 数据指标体系构建是数据分析师的日常工作之一,常见的指标体系方法论包括根据业务发展进程选取由合成略旦易于拆解的指标作为北极星指标。但在实际业务场景中如何运用方法论构建数据指标体系,以监控业务发展呢? 互联网产品按照用户需求进行分类,可以分为工具类、内容类、社交类、交易类以及游戏类。当然,每一个互联网产品并不一定属于单一的某一类别,其类别可能是交叉的。 那各种不同类型的互联网产品都有什么特点?它们对应的北极星指标又分别是什么呢?各类型互联网产品的特点以及北极星指标总结如表1所示。 表 1 各类型互联网产品的特点以及北极星指标 表1 各类型互联网产品的特点以及北极星指标 表1各类型互联网产品的特点以及

C++ AIGC吞吐量测试实战:5个关键指标你必须掌握

第一章:C++ AIGC吞吐量测试的核心意义 在人工智能生成内容(AIGC)系统中,C++作为高性能计算的核心语言之一,承担着大量底层推理与数据处理任务。对C++实现的AIGC模块进行吞吐量测试,是评估系统整体效能的关键环节。吞吐量直接反映单位时间内系统能够处理的请求数量,是衡量服务响应能力、资源利用率和可扩展性的核心指标。 为何吞吐量测试至关重要 * 识别性能瓶颈:通过压力测试发现CPU、内存或I/O层面的限制 * 验证算法优化效果:量化不同实现方案在相同负载下的表现差异 * 支撑容量规划:为集群部署和资源调度提供数据依据 典型测试场景示例 以下是一个基于Google Benchmark框架的C++吞吐量测试代码片段,用于测量文本生成模型单次推理的吞吐能力: #include <benchmark/benchmark.h> // 模拟AIGC模型推理函数 void GenerateText(benchmark::State& state) { for (auto _ : state) { // 模拟一次文本生成过程(实际调用模型推理) std::string re

普通的笔记本电脑使用Faster-Whisper 如何选择模式?

普通的笔记本电脑使用Faster-Whisper 如何选择模式?

CPU 环境下使用 Faster-Whisper 并开启 int8 量化,这几个模型模式(tiny、base、distil-whisper)的主要区别在于识别准确率(WER)、运行速度(RTF)以及对上下文的理解能力。 在 CPU + int8 模式下,你的瓶颈主要在于计算速度和内存带宽。以下是详细的对比分析和建议: 1. 核心区别概览 模型模式参数量速度 (CPU int8)准确率核心优势适用场景Tiny~39M🚀 极快⭐ 基础资源占用极低,响应最快简单的语音指令、极低延迟需求的实时字幕Base~74M⚡ 快⭐⭐ 良好速度与准确率的平衡点日常会议记录、清晰的播客转录Distil-Whisper~756M🐢 较慢⭐⭐⭐⭐ 优秀接近 Large 模型的准确率,抗噪性强复杂口音、背景噪音大、专业术语较多的场景 2. 详细模式解析 🟢 Tiny 模式:极致速度,资源敏感

GitHub Copilot 使用笔记

GitHub Copilot 是 VSCode 自带的 AI Agent 插件,需要登录 GitHub 账号使用,分为免费版和付费版。 关于个人额度,可以在 Github 的 Copilot 菜单里查看 支持模型 添加第三方模型 通过 Manage Models 选中对应厂商。 可以通过 OpenRouter 来导入免费的模型,需要先到 OpenRouter 注册 API Key,输入后即可使用,也可以使用兼容 OpenAI 接口的三方 API,比如 硅基流动 SiliconFlow 使用帮助信息 切换到 Ask 模式,输入 /help 即可获取帮助命令,可以查看当前有什么可用命令和使用方法。 翻译后的内容,方便查看,