Go mod 好菜系列 - 0x02 Gorm 这盘 ORM 到底怎么吃

详细拆 gorm 的定位、模型定义、迁移、CRUD、查询链、关联和事务,看看它为什么常见、又为什么经常引发争议。

只要一聊 Go 数据库生态,gorm 基本很难绕过去。有人觉得它真香,能省掉一大堆 SQL 模板劳动;也有人嫌它太重、太黑盒、容易把查询性能搞成谜语。两边都不算完全错,关键在于你得先知道它是个什么东西。

gorm 是干嘛的

它是一个 ORM。简单说,就是把数据库表和 Go 结构体做了一层映射,让你能更多地用 Go 代码操作数据,而不是每次都从零手写 SQL。

它最典型能帮你做的事包括:

  • 模型定义
  • CRUD
  • 条件查询
  • 关联关系
  • 自动迁移
  • 事务

为什么 gorm 常见

因为很多业务项目都有类似需求:表不少、模型多、增删改查重复劳动多。如果你全手写 SQL,当然更可控,但也确实更容易累。gorm 刚好填了这个“开发效率优先”的位置。

一个最基础的模型

type User struct {
    ID        uint           `gorm:"primaryKey"`
    Name      string         `gorm:"size:64;not null"`
    Email     string         `gorm:"uniqueIndex"`
    CreatedAt time.Time
    UpdatedAt time.Time
}

gorm 默认会根据结构体推表名、推字段,也会自动识别一些常见约定,比如主键、时间字段。

连接数据库

dsn := "user:password@tcp(127.0.0.1:3306)/demo?charset=utf8mb4&parseTime=True&loc=Local"
db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{})
if err != nil {
    return err
}

如果你前面学过 database/sql,这里可以把 gorm 理解成“站在它上面的一层更高抽象”。底层很多事还是那套数据库驱动和连接池逻辑。

自动迁移很好用,但别无脑全信

err := db.AutoMigrate(&User{})
if err != nil {
    return err
}

这一步在开发环境里非常顺手,改个字段就能快速把表结构推过去。但到了正式环境,涉及大表变更、索引调整、数据迁移的时候,别把 AutoMigrate 当神仙按钮。复杂变更还是建议显式管理。

CRUD 写法为什么让人上头

user := User{Name: "Raymond", Email: "ray@example.com"}
if err := db.Create(&user).Error; err != nil {
    return err
}

var result User
if err := db.First(&result, 1).Error; err != nil {
    return err
}

if err := db.Model(&result).Update("name", "Ray").Error; err != nil {
    return err
}

if err := db.Delete(&result).Error; err != nil {
    return err
}

这类写法的优点非常直观:模型感强,样板少,读起来像“我要干什么”,而不是“我要拼哪句 SQL”。

查询链是 gorm 的日常主战场

var users []User
err := db.Where("age > ?", 18).
    Order("created_at desc").
    Limit(20).
    Find(&users).Error

这也是 gorm 让人觉得“效率高”的关键之一。它把很多条件组合写成了链式风格,适合日常业务查询。

关联关系也常见

type User struct {
    ID    uint
    Name  string
    Posts []Post
}

type Post struct {
    ID     uint
    Title  string
    UserID uint
}
var user User
err := db.Preload("Posts").First(&user, 1).Error

Preload 很好用,但也很容易让人不知不觉查出一大堆数据。关联一多,链路一深,性能就开始变玄学。所以能方便,不代表能放飞。

事务也照样能写

err := db.Transaction(func(tx *gorm.DB) error {
    if err := tx.Create(&User{Name: "A"}).Error; err != nil {
        return err
    }
    if err := tx.Create(&User{Name: "B"}).Error; err != nil {
        return err
    }
    return nil
})

这个 API 用起来很顺,而且可读性不错。对很多业务场景来说,这层封装是加分的。

gorm 最常被吐槽什么

  • 生成 SQL 不够透明
  • 复杂查询写着写着还是得掉回原生 SQL
  • 关联多了以后很容易搞出性能问题
  • 团队成员如果不懂底层 SQL,容易“会用但不知后果”

所以 gorm 的正确使用姿势从来不是“我再也不用懂 SQL 了”,而是“我用它提高常规开发效率,但底层 SQL 依然得懂”。

什么时候适合用 gorm

  • 中小型业务系统
  • CRUD 很多、模型很稳的场景
  • 想提高研发效率
  • 团队能接受 ORM 约定

如果你做的是极度复杂、极度性能敏感的查询服务,那 gorm 未必是最优解,甚至你可能最后会混用原生 SQL。

小结

gorm 是一盘很常见的菜,但吃法得对:

  • 它能明显提高常规数据层开发效率
  • 模型、查询链、关联、迁移都很顺手
  • 方便不等于可以不懂 SQL
  • 复杂查询和性能敏感场景别闭眼全靠 ORM

下一篇我们换一道偏基础设施的菜:go-redis。数据库写慢了以后,很多项目下一步都会想碰它。

Read more

Harness Engineering:从驾驭模型到构建 AI 工厂

Harness Engineering:从驾驭模型到构建 AI 工厂

2026 年,AI 编程工具的竞争焦点发生了根本性转变:决定 AI 助手好不好用的,不再是模型本身,而是包裹在模型外面的那层"Harness"。同一个模型,在不同的 Harness 下,性能差距可以达到 78% vs 42%。 这篇文章将带你深入了解 Harness Engineering——这个正在重新定义 AI 工程实践的新兴领域。 一、什么是 Harness Engineering? Harness 的字面意思是"马具/缰绳"——用来驾驭一匹强壮但不受控的马。在 AI 语境下,Harness 就是 LLM 之外的一切:工具定义、记忆系统、权限模型、反馈循环、文档规范、多

By Fuyu Jia

Claude Code CLI + Ralph:让 AI 自动完成大型编程任务的终极方案

TL;DR 当你的编程任务大到一个 AI 对话窗口装不下时,Ralph 会帮你把任务拆成小块,让 Claude Code CLI 一个接一个地自动完成——每轮都用全新的上下文窗口,不会越写越糊涂。 一、什么是 Ralph? Ralph 是一个开源项目(GitHub 16k+ Stars),基于 Geoffrey Huntley 提出的 "Ralph Pattern" 构建。它的核心理念很简单: 不要让 AI 在一个漫长的会话里做完所有事情,而是把大任务拆成小故事,每个故事用一个全新的 AI 实例来完成。 这解决了 AI 编程中最常见的痛点——上下文窗口耗尽。当对话越来越长,AI 的输出质量会明显下降。Ralph 通过「每轮一个新实例」的方式,

By Fuyu Jia