【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)
    }
相关推荐
银发控、6 小时前
MySQL联合索引
数据库·mysql
予枫的编程笔记6 小时前
【MySQL修炼篇】从踩坑到精通:事务隔离级别的3大异常(脏读/幻读/不可重复读)解决方案
数据库·mysql·后端开发·数据库事务·事务隔离级别·rr级别·脏读幻读不可重复读
一起养小猫8 小时前
Flutter for OpenHarmony 实战:记账应用数据统计与可视化
开发语言·jvm·数据库·flutter·信息可视化·harmonyos
世界尽头与你8 小时前
(修复方案)CVE-2023-22047: Oracle PeopleSoft Enterprise PeopleTools 未授权访问漏洞
数据库·安全·oracle·渗透测试
韩立学长8 小时前
【开题答辩实录分享】以《智能大学宿舍管理系统的设计与实现》为例进行选题答辩实录分享
数据库·spring boot·后端
Henry Zhu1238 小时前
数据库(五):反规范化
数据库
Mr_Xuhhh8 小时前
MySQL函数详解:日期、字符串、数学及其他常用函数
java·数据库·sql
he___H9 小时前
Redis高级数据类型
数据库·redis·缓存
霖霖总总9 小时前
[小技巧60]深入解析 MySQL Online DDL:MySQL Online DDL、pt-osc 与 gh-ost 机制与最佳实践
数据库·mysql
爱学习的阿磊9 小时前
使用PyTorch构建你的第一个神经网络
jvm·数据库·python