测试Go结构体方法需使用testing包,通过创建实例调用方法并验证返回值或状态变化。1. 使用Test开头函数和t.Run组织子测试;2. 检查方法对字段的修改;3. 表驱动测试覆盖多场景;4. 依赖接口与mock实现隔离外部服务,确保测试准确性和可维护性。

测试 Go 中结构体方法的正确性,主要依赖 Go 的 testing 包 和清晰的测试用例设计。核心思路是:创建结构体实例,调用其方法,验证返回值或状态变化是否符合预期。
1. 基本测试结构
使用标准库 testing 编写测试函数,函数名以 Test 开头,参数为 *testing.T。通过 t.Run 可组织子测试,提高可读性。
func TestUser_GetFullName(t *testing.T) {
user := User{Name: "Alice", Surname: "Smith"}
fullName := user.GetFullName()
if fullName != "Alice Smith" {
t.Errorf("期望 'Alice Smith',实际 '%s'", fullName)
}
}
2. 验证方法对内部状态的影响
某些方法会改变结构体字段,测试时需检查调用前后字段值的变化。
func TestAccount_Deposit(t *testing.T) {
acc := &Account{Balance: 100}
acc.Deposit(50)
if acc.Balance != 150 {
t.Errorf("余额应为150,实际为%d", acc.Balance)
}
}
3. 使用表驱动测试覆盖多种情况
定义测试用例切片,包含输入、期望输出和描述,适用于有明确输入输出的方法。
立即学习“go语言免费学习笔记(深入)”;
func TestCalculator_Add(t *testing.T) {
calc := Calculator{}
tests := []struct{
a, b int
expected int
desc string
}{
{2, 3, 5, "正数相加"},
{0, 0, 0, "零值测试"},
{-1, 1, 0, "负数与正数"},
}
for _, tc := range tests {
t.Run(tc.desc, func(t *testing.T) {
result := calc.Add(tc.a, tc.b)
if result != tc.expected {
t.Errorf("Add(%d,%d) = %d, 期望 %d",
tc.a, tc.b, result, tc.expected)
}
})
}
}
4. 模拟依赖与接口隔离
若方法依赖外部服务(如数据库、HTTP),应将依赖抽象为接口,并在测试中使用模拟实现。
例如:
type Notifier interface {
Send(message string) error
}
func (u *User) Notify(notifier Notifier) error {
return notifier.Send("Hello " + u.Name)
}
// 测试时使用 mock
type MockNotifier struct {
Called bool
Msg string
}
func (m *MockNotifier) Send(msg string) error {
m.Called = true
m.Msg = msg
return nil
}
func TestUser_Notify(t *testing.T) {
user := User{Name: "Bob"}
mock := &MockNotifier{}
user.Notify(mock)
if !mock.Called {
t.Error("期望调用 Send")
}
if mock.Msg != "Hello Bob" {
t.Errorf("消息内容错误: %s", mock.Msg)
}
}
基本上就这些。关键在于构造真实场景的实例,调用方法后准确断言结果,结合表驱动和 mock 提升覆盖率和可靠性。










