【Go】新版GORM自动字段映射规则

新版 GORM(v2 及以上)的自动字段映射遵循一套清晰的规则,主要围绕结构体字段名到数据库列名的转换 ,以及特殊字段的默认处理,具体规则如下:

1. 命名转换规则(核心)

GORM 会将结构体的驼峰式命名(CamelCase) 自动转换为数据库的蛇形命名(snake_case),这是最常用的映射规则。

  • 转换逻辑:在字段名的大写字母前添加下划线,并将所有字母转为小写。
  • 示例:
    • 结构体字段 StudentId → 数据库列 student_id
    • 结构体字段 ScoreValue → 数据库列 score_value
    • 结构体字段 CreatedAt → 数据库列 created_at
    • 结构体字段 ID → 数据库列 id(特殊处理:连续大写会被合并转换)

2. 特殊字段的默认映射

GORM 对一些特定名称的结构体字段有内置映射规则,无需额外配置:

  • ID:默认作为主键(primaryKey),映射到数据库列 id
  • CreatedAt:自动映射到 created_at 列,GORM 会在插入记录时自动填充当前时间。
  • UpdatedAt:自动映射到 updated_at 列,GORM 会在更新记录时自动更新为当前时间。
  • DeletedAt:如果结构体包含该字段,GORM 会启用"软删除"功能,映射到 deleted_at 列(未删除时为 NULL,删除时填充时间)。

3. 自定义映射(覆盖默认规则)

如果需要自定义列名(不遵循驼峰转蛇形),可以通过 gorm:"column:自定义列名" 标签显式指定,覆盖默认规则。

示例:

go 复制代码
type Score struct {
    Id         uint64    `gorm:"primaryKey" json:"id"`  // 默认映射为 id
    StudentId  uint64    `gorm:"column:stu_id" json:"student_id"`  // 自定义为 stu_id
    Subject    string    `gorm:"column:course_name" json:"subject"`  // 自定义为 course_name
    // ...
}

4. 忽略字段

如果某些结构体字段不需要映射到数据库列,可以通过 gorm:"-" 标签排除:

go 复制代码
type Score struct {
    // ...
    TempData string `gorm:"-" json:"temp_data"`  // 不会生成数据库列
}

5. 自定义映射的核心方式:配置 NamingStrategy

新版 GORM 支持自定义自动映射规则,可以通过配置 NamingStrategy 来修改默认的字段名、表名映射逻辑,满足特定的命名规范需求。

GORM 提供了 gorm.NamingStrategy 结构体,允许你自定义:

  • 表名映射规则(结构体名 → 表名)
  • 字段名映射规则(结构体字段名 → 数据库列名)
  • 联表外键命名规则
  • 索引命名规则

通过在初始化 GORM 时配置 NamingStrategy 即可生效。

示例:自定义字段名映射规则

默认规则是「驼峰转蛇形」,如果需要修改(例如改为全大写、添加前缀等),可以自定义 Column 函数:

go 复制代码
import (
  "gorm.io/driver/sqlite"
  "gorm.io/gorm"
  "strings"
)

// 自定义命名策略
namingStrategy := gorm.NamingStrategy{
  // 自定义字段名映射:将驼峰转为全大写(而非蛇形)
  Column: func(column string) string {
    // 示例:StudentId → STUDENTID
    return strings.ToUpper(column)
  },
  // 可选:自定义表名映射(结构体名 → 表名)
  Table: func(table string) string {
    // 示例:在表名前加前缀 "t_"
    return "t_" + table
  },
}

// 初始化 GORM 时应用命名策略
db, err := gorm.Open(sqlite.Open("test.db"), &gorm.Config{
  NamingStrategy: namingStrategy,
})

上述配置后:

  • 结构体字段 StudentId 会映射为数据库列 STUDENTID(而非默认的 student_id
  • 结构体 Score 会映射为表 t_score(而非默认的 scores

更复杂的自定义示例(保持部分默认逻辑)

如果只想在默认规则基础上微调(例如给所有字段名加前缀),可以结合默认转换逻辑:

go 复制代码
import (
  "gorm.io/gorm/schema"
  "strings"
)

namingStrategy := gorm.NamingStrategy{
  Column: func(column string) string {
    // 先使用默认的驼峰转蛇形,再添加前缀 "col_"
    defaultColumn := schema.NamingStrategy{}.Column(column) // 调用默认转换
    return "col_" + defaultColumn
  },
}

效果:

  • 结构体字段 ScoreValue → 默认转换为 score_value → 最终映射为 col_score_value

注意事项

  1. 自定义规则全局生效,会影响所有结构体的映射。

  2. 若个别字段需要特殊处理,可以用 gorm:"column:xxx" 标签单独指定,优先级高于全局规则。

  3. 表名默认会在结构体名基础上做「驼峰转蛇形 + 复数化」(如 Userusers),可通过 SingularTable: true 禁用复数化:

    go 复制代码
    namingStrategy := gorm.NamingStrategy{
      SingularTable: true, // 表名不加复数后缀(User → user)
    }
相关推荐
2501_941805937 小时前
在大阪智能零售场景中构建支付实时处理与高并发顾客行为分析平台的工程设计实践经验分享
数据库
李慕婉学姐7 小时前
【开题答辩过程】以《基于JAVA的校园即时配送系统的设计与实现》为例,不知道这个选题怎么做的,不知道这个选题怎么开题答辩的可以进来看看
java·开发语言·数据库
珠海西格电力7 小时前
零碳园区有哪些政策支持?
大数据·数据库·人工智能·物联网·能源
数据大魔方8 小时前
【期货量化实战】日内动量策略:顺势而为的短线交易法(Python源码)
开发语言·数据库·python·mysql·算法·github·程序员创富
Chasing Aurora8 小时前
数据库连接+查询优化
数据库·sql·mysql·prompt·约束
倔强的石头_8 小时前
【金仓数据库】ksql 指南(六)—— 创建与管理用户和权限(KingbaseES 安全控制核心)
数据库
小熊officer9 小时前
Python字符串
开发语言·数据库·python
渐暖°10 小时前
JDBC直连ORACLE进行查询
数据库·oracle
萧曵 丶10 小时前
Next-Key Lock、记录锁、间隙锁浅谈
数据库·sql·mysql·mvcc·可重复读·幻读
做cv的小昊11 小时前
【TJU】信息检索与分析课程笔记和练习(7)数据库检索—Ei
数据库·笔记·学习·全文检索