Golang测试表格驱动与泛型结合示例

答案:Go中结合泛型与表格驱动测试可提升代码复用和类型安全,通过泛型断言函数assertEqual和通用测试结构体testFindIndexGeneric,减少重复逻辑,支持多类型验证,使单元测试更简洁高效。

golang测试表格驱动与泛型结合示例

在 Go 语言中,表格驱动测试(Table-Driven Tests)是编写单元测试的常见模式。结合泛型,可以构建更通用、类型安全的测试逻辑,减少重复代码。下面是一个将泛型与表格驱动测试结合使用的实用示例。

使用泛型简化断言逻辑

我们定义一个泛型辅助函数来比较期望值和实际值,避免每个测试用例都写重复的判断逻辑。

func assertEqual[T comparable](t *testing.T, name string, got, want T) {    t.Helper()    if got != want {        t.Errorf("[%s] expected: %v, got: %v", name, want, got)    }}

这个 assertEqual 函数适用于任何可比较类型(如 int、string、struct 等),让测试更简洁。

为泛型函数编写表格测试

假设我们有一个泛型查找函数 FindIndex,它在切片中查找满足条件的第一个元素索引:

立即学习“go语言免费学习笔记(深入)”;

func FindIndex[T any](slice []T, predicate func(T) bool) int {    for i, v := range slice {        if predicate(v) {            return i        }    }    return -1}

我们可以为它编写表格驱动测试,覆盖多种类型场景:

func TestFindIndex(t *testing.T) {    tests := []struct {        name     string        slice    interface{}        pred     interface{}        want     int    }{        {            name:  "int: 找到偶数",            slice: []int{1, 3, 4, 5},            pred:  func(x int) bool { return x%2 == 0 },            want:  2,        },        {            name:  "string: 找到空字符串",            slice: []string{"a", "", "b"},            pred:  func(s string) bool { return s == "" },            want:  1,        },        {            name:  "struct: 找到特定字段",            slice: []Person{{"Alice", 25}, {"Bob", 30}},            pred:  func(p Person) bool { return p.Name == "Bob" },            want:  1,        },        {            name:  "未找到",            slice: []int{1, 2, 3},            pred:  func(x int) bool { return x > 10 },            want:  -1,        },    }    for _, tt := range tests {        t.Run(tt.name, func(t *testing.T) {            switch slice := tt.slice.(type) {            case []int:                pred := tt.pred.(func(int) bool)                got := FindIndex(slice, pred)                assertEqual(t, tt.name, got, tt.want)            case []string:                pred := tt.pred.(func(string) bool)                got := FindIndex(slice, pred)                assertEqual(t, tt.name, got, tt.want)            case []Person:                pred := tt.pred.(func(Person) bool)                got := FindIndex(slice, pred)                assertEqual(t, tt.name, got, tt.want)            }        })    }}type Person struct {    Name string    Age  int}

虽然这里用了 interface{} 存储不同类型,但通过类型断言确保类型安全。测试结构清晰,易于扩展新类型。

进一步优化:使用泛型测试结构体

如果只测试单一类型,可以直接使用泛型结构体,避免类型断言:

func testFindIndexGeneric[T comparable](t *testing.T, name string, slice []T, pred func(T) bool, want int) {    t.Run(name, func(t *testing.T) {        got := FindIndex(slice, pred)        assertEqual(t, name, got, want)    })}func TestFindIndex_GenericHelper(t *testing.T) {    testFindIndexGeneric(t, "整数查找", []int{10, 20, 30}, func(x int) bool { return x > 15 }, 1)    testFindIndexGeneric(t, "字符串查找", []string{"go", "rust", "ts"}, func(s string) bool { return s == "rust" }, 1)}

这种方式更安全、更简洁,适合类型明确的测试场景。

基本上就这些。泛型 + 表格驱动能让 Go 测试更灵活、少冗余,关键是设计好可复用的辅助函数和结构。不复杂但容易忽略细节类型匹配。

以上就是Golang测试表格驱动与泛型结合示例的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1412635.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月16日 06:30:33
下一篇 2025年12月16日 06:30:47

相关推荐

  • C++ 函数模板如何提高代码可复用性?

    使用函数模板来提升 c++++ 代码的可复用性,可根据不同数据类型提供泛化的函数。优点包括:代码可复用性:消除重复创建特定类型函数的需要。类型安全性:编译器确保函数仅用于兼容的值。灵活性:可用于处理不同类型的函数,提供更大的灵活性。 利用 C++ 函数模板提升代码可复用性 函数模板是一种强大的 C+…

    2025年12月18日
    000
  • Lambda 表达式在 C++ 函数中的面向对象设计

    在 c++++ 中,lambda 表达式为面向对象设计提供了灵活的扩展方式,可用于实现事件处理程序、回调函数、过滤器和比较器。它们的基本语法为:[capture list] (parameters) -> return type { body },其中 capture list 指定外部变量访…

    2025年12月18日
    000
  • C++ 函数如何指定自定义枚举类型作为返回类型

    c++++ 中可使用自定义枚举类型作为函数返回类型,从而返回一组预定义可能值之一。实现步骤如下:创建自定义枚举类型,使用 enum 关键字和枚举名称及值列表。指定函数返回类型,使用枚举类型名称作为返回类型。 使用自定义枚举类型作为 C++ 函数返回类型 在 C++ 中,我们可以使用枚举类型作为函数的…

    2025年12月18日
    000
  • C++ 函数的艺术:泛型编程探究,代码复用与抽象

    c++++泛型编程允许您编写可在各种数据类型上工作的代码,提高代码复用性并减少错误:泛型函数: 使用模板定义,采用类型参数,可操作任何类型的输入;模板特化: 为某些数据类型提供不同的实现,例如浮点数的特殊处理;stl 容器: 标准模板库提供泛型容器,用于类型安全且高效的数据存储和操作;泛型排序算法:…

    2025年12月18日
    000
  • C++ 框架中代码复用和抽象的最佳实践

    最佳实践:代码复用:使用模板和泛型编程。利用继承和派生。提取代码到函数。创建库。使用第三方库。抽象:定义抽象基类或接口。使用虚函数。创建抽象工厂。利用模板元编程(tmp)。应用设计模式。 C++ 框架中代码复用和抽象的最佳实践 引言代码复用和抽象是编写可维护、可扩展且易于测试 C++ 框架的关键技术…

    2025年12月18日
    000
  • 如何在C++应用程序中使用框架实现代码复用?

    通过使用框架,可以在 c++++ 应用程序中实现代码复用。具体步骤如下:选择适用于特定需求的框架(如 qt、mfc、boost)。集成框架,包括包含头文件和使用框架对象。实战案例中,可以使用 qt 创建 gui 应用程序,使用 mfc 创建文件浏览器窗口。 如何在 C++ 应用程序中使用框架实现代码…

    2025年12月18日
    000
  • 使用 C++ 框架实现松散耦合和代码复用

    c++++ 框架通过以下机制实现松散耦合和代码复用:接口:定义组件交互的公共契约,实现松散耦合。抽象类:提供公共实现,不同实现通过继承获得通用接口。模板:创建泛型函数和类,适用于各种类型,实现代码复用。继承:共享父类中已存在的代码。 利用 C++ 框架实现松散耦合和代码复用 简介 在大型软件系统中,…

    2025年12月18日
    000
  • C++ 框架如何在跨平台开发中实现代码重用性?

    c++++框架通过提供以下特性促进跨平台开发中的代码重用性:可重用的组件:提供可用于构建应用程序的类、函数和库等组件。代码抽象:抽象底层平台差异,提供跨平台api,以便使用一致的接口访问不同的操作系统和平台上的功能。 C++ 框架:跨平台开发中的代码重用性利器 导言 跨平台开发意味着开发可以在多个操…

    2025年12月18日
    000
  • C++ 框架设计中实现代码重用的技术

    在 c++++ 框架设计中,实现代码重用的技术包括:模板方法模式:定义算法大纲,由子类定义具体步骤。策略模式:分离算法实现和使用对象,提高灵活性。工厂方法模式:创建对象的方法由子类实现,允许创建不同类型对象。抽象工厂模式:创建相关对象家族的方法,无需指定具体类,促进松耦合。单例模式:确保类只有一个实…

    2025年12月18日
    000
  • 剖析 C++ 函数指针增强代码复用能力的原理

    函数指针是一种指向函数的指针,允许动态调用函数,从而增强代码复用性。例如,可创建一个通用折扣计算函数,接受函数指针作为参数,并为不同折扣类型创建不同的函数,通过传递不同的函数指针实现不同折扣计算。在 c++++ 中,排序策略函数指针可用于根据排序策略对学生列表排序,展示函数指针在代码复用中的应用。 …

    2025年12月18日
    000
  • C++ 中的泛型算法是如何复用功能的?

    c++++ 泛型算法可复用通用操作,包括:排序算法(如 sort)搜索算法(如 find)集合操作(如 set_difference)转换算法(如 transform)使用泛型算法时,需提供输入容器、输出容器(可选)和 function 对象作为参数。例如,sort 算法可用于整数数组排序。自定义比…

    2025年12月18日
    000
  • C++ 中的泛型容器是如何实现代码复用的?

    泛型容器是 c++++ 中可容纳各种数据类型的容器,使用模板机制实现。通过模板创建,可以使用任何类型的元素。消除特定类型容器创建需要,实现代码复用。在数据结构库、数据库、缓存系统等领域广泛应用。优势:代码复用、类型安全、性能优化。注意:非必须容器泛型,可扩展性会增加代码和内存开销。 C++ 中的泛型…

    2025年12月18日
    000
  • 设计模式提升代码复用性的技巧和方法

    设计模式提升了代码可重用性,提供了可重复使用的结构,可通过抽象化对象创建、封装实现和松耦合等方式实现:1. 工厂模式简化对象创建,使您可以无缝替换和组装对象;2. 抽象工厂模式将创建对象家族的职责从客户端代码中分离出来;3. 桥接模式解耦了抽象和实现,允许独立更改;4. 单例模式确保只有一个实例,提…

    2025年12月18日
    000
  • 模板化编程能带来什么好处?

    模板化编程可提升代码质量,因为它:增强可读性:封装重复代码,使其更易理解。提升可维护性:只需更改模板即可适应数据类型变更。优化效率:编译器生成特定数据类型的优化代码。促进代码复用:创建通用的算法和数据结构,可重复使用。 模板化编程的力量:提升代码可读性、可维护性和效率 简介 模板化编程是一种高级编程…

    2025年12月18日
    000
  • C语言中go out的用法详解

    在C语言中,”go out”是一个常用的术语,指的是函数的退出和返回值的传递。在本文中,我们将详细解释C语言中”go out”的用法,并提供具体的代码示例。 在C语言中,函数的返回值通过return语句传递给调用函数。return语句用于终止函数的执行…

    2025年12月17日
    000
  • C# Avalonia如何集成Entity Framework Core Avalonia EF Core教程

    在 Avalonia 中集成 EF Core 可行,关键在于异步操作、DI 注入 DbContextFactory 及正确管理生命周期;需避免 UI 线程阻塞,推荐用 AddDbContextFactory 而非 Scoped 或 Singleton 注册。 在 Avalonia 中集成 Entit…

    2025年12月17日
    000
  • Blazor 导航时通过URL传递参数的方法

    Blazor导航传参主要通过路由模板实现:路径参数(如@page “/counter/{id:int}”)用于必填标识性数据,自动绑定到[Parameter]属性;查询参数需手动解析,适合非必需或动态参数;NavLink仅支持字符串插值传路径参数。 Blazor 中导航时通过…

    2025年12月17日
    000
  • MAUI怎么打包安卓应用 MAUI APK打包发布教程

    MAUI打包安卓APK需四步:改格式为apk、配置AndroidManifest.xml权限与基础信息、通过发布流程生成、添加签名。缺一将导致无法安装或闪退,签名密钥须备份以防更新失败。 MAUI 打包安卓 APK 不难,但几个关键步骤漏掉一个,就装不上或一启动就闪退。核心就四步:改格式、配权限、打…

    2025年12月17日
    000
  • Dapper怎么处理多对多关系 Dapper many-to-many查询映射

    Dapper通过手动JOIN中间表+MultiMapping+字典缓存实现多对多映射,核心是SQL扁平查询、splitOn分割字段、内存重组对象树;需注意LEFT JOIN处理空关联、字段别名防冲突、集合初始化及大数据量性能优化。 Dapper 本身不自动处理多对多关系,但通过手动编写连接查询 + …

    2025年12月17日
    000
  • Blazor 共享布局 MasterPage 设置方法

    Blazor 使用 Layout 组件替代 ASP.NET Web Forms 的 MasterPage,功能一致且更灵活;通过继承 LayoutComponentBase、定义 @Body 占位符、支持依赖注入与嵌套布局实现解耦式 UI 结构。 Blazor 没有传统 ASP.NET Web Fo…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信