Go测试中必须mock数据库而非连真实库,因真实库会拖慢速度、引入环境依赖、导致数据污染和随机失败;sqlmock是主流方案,通过包装sql.Driver模拟SQL行为,支持断言SQL、参数检查和事务模拟,且需调用ExpectationsWereMet()验证。
真实数据库会拖慢测试速度、引入环境依赖、导致测试间数据污染,还可能因网络或权限问题让 go test 随机失败。尤其 CI 环境中,你没法保证 MySQL 实例一定在 localhost:3306 且账号有写权限。
所以必须 mock:不是“要不要”,而是“怎么 mock 得像、不漏测、不难维护”。
sqlmock 是目前 Go 生态中最主流的 SQL 层 mock 方案,它不替换 *sql.DB,而是包装一个假的 sql.Driver,让 sql.Open("sqlmock", "") 返回可控的 *sql.DB,所有 Query/Exec 调用都走预设规则。
database/sql 使用习惯,不用改业务代码mock.ExpectQuery(`^SELECT.*FROM users`)),也能检查参数绑定值mock.ExpectBegin() + mock.ExpectCommit() 或 ExpectRollback()
mock.ExpectationsWereMet(),否则未触发的期望会报错 —— 这是新手最常漏的一步示例片段:
db, mock, _ := sqlmock.New()
defer db.Close()
mock.ExpectQuery(`SELECT id FROM users WHERE name = \?`).WithArgs("alice").WillReturnRows(
sqlmock.NewRows([]string{"id"}).AddRow(123),
)
rows, _ := db.Query("SELECT id FROM users WHERE name = ?", "alice")
// ... 处理 rows
if err := mock.ExpectationsWereMet(); err != nil {
t.Fatalf("unfulfilled expectations: %v", err)
}
有人图省事用 sqlite3 的 :memory: 数据库跑测试,看似隔离,实则隐患不少:
INSERT IGNORE、PostgreSQL 的 ON CONFLICT 在 SQLite 里不生效或行为不同:memory: 数据库不是进程级隔离,多个 test case 可能互相干扰它适合做集成测试前的快速 smoke check,但不该替代 sqlmock 做单元测试。

GORM 自带 gorm.io/gorm/migrator 和底层 dialector,直接 mock *gorm.DB 很容易掉进反射和链式调用陷阱。推荐做法是:
UserRepository.FindByID)做接口抽象,然后 mock 接口,而非 mock GORM 实例本身sqlmock 包住 GORM 底层的 *sql.DB,GORM 会自动使用它 —— 设置方式:gorm.Open(mysql.Open(db), &gorm.Config{}),其中 db 是 sqlmock.New() 返回的gorm.Session 或 Scopes:它们生命周期短、内部状态复杂,mock 成本远高于收益关键点在于:mock 的边界要落在“你真正想验证的行为”上。如果函数只是封装了 db.Where(...).First(),那就 mock db;如果函数逻辑含复杂条件拼接,就该把拼 SQL 的部分抽成独立函数再单独测。
真正难的不是选哪个库,而是决定哪一层该被 mock、哪一层该被真实调用 —— 这取决于你这次测试想回答的问题:是“SQL 写对了吗”,还是“事务控制是否可靠”,或是“错误路径是否被覆盖”。没想清楚这点,mock 就容易变成测试套娃。