golang通过AutoMigrate方法自动创建table详解

一.AutoMigrate介绍

1.介绍

在 Go 语言中,GORM支持Migration特性,支持根据Go Struct结构自动生成对应的表结构,使用 GORM ORM 库的 AutoMigrate 方法可以自动创建数据库表,确保数据库结构与定义的模型结构一致。AutoMigrate 方法非常方便,特别适合在开发阶段进行快速迭代

注意: 

  • AutoMigrate 会创建表、缺失的外键、约束、列和索引
  • 出于保护数据的目的,它 不会 删除未使用的列
  • AutoMigrate 会自动创建数据库外键约束,可以在初始化时禁用此功能
db, err := gorm.Open(sqlite.Open("gorm.db"), &gorm.Config{DisableForeignKeyConstraintWhenMigrating: true,
})
  • 官网地址:迁移 | GORM - AutoMigrate

2.AutoMigrate 的工作原理

  • 创建表:如果表不存在,AutoMigrate 会创建它
  • 更新表结构:如果表已经存在,AutoMigrate 会检查结构体中字段的变化,例如新字段、字段类型的更改等,并进行相应的更新
  • 不删除字段AutoMigrate 不会删除现有字段或表。如果需要删除字段,你需要手动执行此操作,或者使用 GORM 的 Migrate 功能来编写复杂的迁移逻辑

二.AutoMigrate使用

1.模型定义

模型是标准的 struct,由 Go 的基本数据类型、实现了 Scanner 和 Valuer 接口的自定义类型及其指针或别名组成,如:

package modelimport ("encoding/json""fmt""strings""time""gorm.io/gorm"
)func InitSwitch(dbClient *gorm.DB) {err := dbClient.AutoMigrate(&Switch{})if err != nil {panic(fmt.Sprintf("migrate table agent fail: %v", err))}
}type Switch struct {Id        uint64    `gorm:"primaryKey;autoIncrement;comment:开关ID" json:"id"`                                                           // 开关IDKey       string    `gorm:"column:key;type:varchar(255) not null;comment:Key;uniqueIndex:key"`                                                         // keyValue     string    `gorm:"column:value;type:varchar(512) not null;comment:值: 1001|1,1002|1,1003|0; eg: 1001|1 表示:1001-渠道包,1 是否开启该功能"` // 值: 1001|1,1002|1,1003|0; eg: 1001|1 表示:1001-渠道包,1 是否开启该功能Content   string    `gorm:"column:content;type:varchar(512) not null;comment:内容说明"`                                                    // 内容说明Name      string    `gorm:"column:name;type:varchar(255) not null;comment:名称"`                                                         // 名称Type      uint64    `gorm:"column:type;comment:类型: 1 功能开关"`                                                                            // 类型: 1 功能开关CreatedAt time.Time `json:"created_at"`                                                                                                //  创建时间UpdatedAt time.Time `json:"updated_at"`                                                                                                //  更新时间
}func (s *Switch) TableName() string {return "switch_info"
}

2.字段标签

 声明 model 时,tag 是可选的,GORM支持以下 tag,标签之间用";"隔开: tag 名大小写不敏感,但建议使用 camelCase 风格

 3.自动建表

通过AutoMigrate函数可以快速建表,如果表已经存在不会重复创

// 根据User结构体,自动创建表结构.
db.AutoMigrate(&Switch{})// 一次创建User、Switch、Order三个结构体对应的表结构
db.AutoMigrate(&User{}, &Switch{}, &Order{})// 可以通过Set设置附加参数,下面设置表的存储引擎为InnoDB
db.Set("gorm:table_options", "ENGINE=InnoDB").AutoMigrate(&Switch{})

使用AutoMigrate自动创建表:实际使用只需要调用InitSwitch方法即可:

package mainimport ("model"
)
func main() {// 初始化Switch,自动创建表model.InitSwitch(*gorm.DB)
}

当调用InitSwtich方法后,系统会自动地在对应的数据库下面创建响应的表,这样就可以根据实际情况修改对应的结构体,来实现表的自定义创建: 

 

三.Schema方法

在使用AutoMigrate方法创建表时,也许会用到其他的方法,这里列举了一些常见的,如下:

1.检测表是否存在

// 检测User结构体对应的表是否存在
db.Migrator().HasTable(&User{})// 检测表名users是否存在
db.Migrator().HasTable("users")

2.建表

// 根据User结构体建表
db.Migrator().CreateTable(&User{})

3.删除表

// 删除User结构体对应的表
db.Migrator().DropTable(&User{})// 删除表名为users的表
db.Migrator().DropTable("users")

4.删除字段

// 删除User结构体对应表中的description字段
db.Migrator().DropColumn(&User{}, "Name")

5.添加索引

type User struct {gorm.ModelName string `gorm:"size:255;index:idx_name,unique"`
}// 为 Name 字段创建索引
db.Migrator().CreateIndex(&User{}, "Name")
db.Migrator().CreateIndex(&User{}, "idx_name")// 为 Name 字段删除索引
db.Migrator().DropIndex(&User{}, "Name")
db.Migrator().DropIndex(&User{}, "idx_name")// 检查索引是否存在
db.Migrator().HasIndex(&User{}, "Name")
db.Migrator().HasIndex(&User{}, "idx_name")type User struct {gorm.ModelName  string `gorm:"size:255;index:idx_name,unique"`Name2 string `gorm:"size:255;index:idx_name_2,unique"`
}
// 修改索引名
db.Migrator().RenameIndex(&User{}, "Name", "Name2")
db.Migrator().RenameIndex(&User{}, "idx_name", "idx_name_2")

6.组合索引

type User struct {Name   string `gorm:"index:idx_member"`Number string `gorm:"index:idx_member"`
}

四.注意事项

  • 数据丢失风险:在生产环境中使用 AutoMigrate 时要小心,特别是字段类型的变更可能会导致数据丢失。建议在生产环境中使用手动迁移
  • 复杂的迁移逻辑:如果需要进行更复杂的迁移逻辑(如添加索引、修改列类型等),可以使用 GORM 的 Migrator() 方法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/68570.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PHP If...Else 语句详解

PHP If...Else 语句详解 引言 在PHP编程中,if...else语句是流程控制的重要组成部分,它允许程序根据条件判断执行不同的代码块。本文将详细解析PHP中的if...else语句,包括其基本用法、高级技巧以及注意事项。 一、基本用法 if...else语句的…

宝塔mysql数据库容量限制_宝塔数据库mysql-bin.000001占用磁盘空间过大

磁盘空间占用过多,排查后发现网站/www/wwwroot只占用7G,/www/server占用却高达8G,再深入排查发现/www/server/data目录下的mysql-bin.000001和mysql-bin.000002两个日志文件占去了1.5G空间。 百度后学到以下知识,做个记录。 mysql…

【Leetcode 每日一题】119. 杨辉三角 II

问题背景 给定一个非负索引 r o w I n d e x rowIndex rowIndex,返回「杨辉三角」的第 r o w I n d e x rowIndex rowIndex 行。 在「杨辉三角」中,每个数是它左上方和右上方的数的和。 数据约束 0 ≤ r o w I n d e x ≤ 33 0 \le rowIndex \le 33 …

Case逢无意难休——深度解析JAVA中case穿透问题

Case逢无意难休——深度解析JAVA中case穿透问题~ 不作溢美之词,不作浮夸文章,此文与功名进取毫不相关也!与大家共勉!! 更多文章:个人主页 系列文章:JAVA专栏 欢迎各位大佬来访哦~互三必回&#…

17、Spring MVC 框架:构建强大的 Java Web 应用程序

嘿,Java 开发者们!今天我们将深入探讨 Spring MVC 框架,它是 Spring 框架中专门用于构建 Web 应用程序的一个强大模块。Spring MVC 遵循经典的 MVC(Model-View-Controller)设计模式,让我们能够轻松地开发出…

decison tree 决策树

熵 信息增益 信息增益描述的是在分叉过程中获得的熵减,信息增益即熵减。 熵减可以用来决定什么时候停止分叉,当熵减很小的时候你只是在不必要的增加树的深度,并且冒着过拟合的风险 决策树训练(构建)过程 离散值特征处理:One-Hot…

Ubuntu 手动安装 Open WebUI 完整指南

Ubuntu 手动安装 Open WebUI 完整指南 前提条件 在安装 Open WebUI 之前,请确保您的系统满足以下要求: Ubuntu 22.04 LTS 或更高版本Python 3.10Node.js 18Git至少 4GB 内存足够的磁盘空间(推荐 20GB 以上) 安装步骤 1. 更新…

研发的立足之本到底是啥?

0 你的问题,我知道! 本文深入T型图“竖线”的立足之本:专业技术 技术赋能业务能力。研发在学习投入精力最多,也误区最多。 某粉丝感发展遇到瓶颈,项目都会做,但觉无提升,想跳槽。于是&#x…

WPF基础 | 深入 WPF 事件机制:路由事件与自定义事件处理

WPF基础 | 深入 WPF 事件机制:路由事件与自定义事件处理 一、前言二、WPF 事件基础概念2.1 事件的定义与本质2.2 常见的 WPF 事件类型 三、路由事件3.1 路由事件的概念与原理3.2 路由事件的三个阶段3.3 路由事件的标识与注册3.4 常见的路由事件示例 四、自定义事件处…

DeepSeekMoE:迈向混合专家语言模型的终极专业化

一、结论写在前面 论文提出了MoE语言模型的DeepSeekMoE架构,目的是实现终极的专家专业化(expert specialization)。通过细粒度的专家分割和共享专家隔离,DeepSeekMoE相比主流的MoE架构实现了显著更高的专家专业化和性能。从较小的2B参数规模开始&#x…

机器人抓取与操作经典规划算法(深蓝)——2

1 经典规划算法 位姿估计:(1)相机系位姿 (2)机器人系位姿 抓取位姿:(1)抓取位姿计算 (2)抓取评估和优化 路径规划:(1)笛卡…

Linux 内核中的高效并发处理:深入理解 hlist_add_head_rcu 与 NAPI 接口

在 Linux 内核的开发中,高效处理并发任务和数据结构的管理是提升系统性能的关键。特别是在网络子系统中,处理大量数据包的任务对性能和并发性提出了极高的要求。本文将深入探讨 Linux 内核中的 hlist_add_head_rcu 函数及其在 NAPI(网络接收处理接口)中的应用,揭示这些机制…

【Qt】06-对话框

对话框 前言一、模态和非模态对话框1.1 概念1.2 模态对话框1.2.1 代码QAction类 1.2.2 模态对话框运行分析 1.3 非模态对话框1.3.1 代码局部变量和成员变量setAttribute 类 1.3.2 现象解释 二、标准对话框2.1 提示对话框 QMessageBox2.1.1 现象及解释 2.2 问题对话框2.2.1 现象…

< OS 有关 > Android 手机 SSH 客户端 app: connectBot

connectBot 开源且功能齐全的SSH客户端,界面简洁,支持证书密钥。 下载量超 500万 方便在 Android 手机上,连接 SSH 服务器,去运行命令。 Fail2ban 12小时内抓获的 IP ~ ~ ~ ~ rootjpn:~# sudo fail2ban-client status sshd Status for the jail: sshd …

pyserial和pymodbus

pyserial和pymodubus pyserial 和 pymodbus 是两个常用的 Python 库,分别用于串行通讯和 Modbus 协议通讯,广泛应用于与硬件设备(如传感器、PLC、RS-485 转串行设备等)进行数据交换。 pyserial pyserial 是一个用于与串行端口&…

Vue.js `setup()` 函数的使用

Vue.js setup() 函数的使用 今天我们来聊聊 Vue 3 中的 setup() 函数。如果你正在使用 Vue 3,那么对 setup() 函数的理解和掌握将对你的开发工作大有裨益。 什么是 setup() 函数? setup() 函数是 Vue 3 组合式 API(Composition API&#x…

【某大厂一面】HashSet底层怎么实现的

HashSet 是 Java 集合框架中的一个非常常用的集合类,它实现了 Set 接口,并且底层通常是通过 哈希表(HashMap)来实现的。要理解 HashSet 的底层实现,我们需要从哈希表的工作原理开始讲起。下面是对 HashSet 底层实现的详…

是否需要显式使用 epoll_ctl ( fd , EPOLL_CTL_DEL , ... ) 来从红黑树里显式删除过期的套接字

(1)关于 epoll 操作的三大系统函数: epoll_create ( … ) 、 epoll_ctl ( … ) 、 epoll_wait(…)。具体的函数原型略。这些函数完成了 epoll 对象的创建、 套接字往 epoll 红黑树中的添加、修改 与 删除 。 本文的整理…

算法随笔_30: 去除重复字母

上一篇:算法随笔_29:最大宽度坡_方法3-CSDN博客 题目描述如下: 给你一个字符串 s ,请你去除字符串中重复的字母,使得每个字母只出现一次。需保证 返回结果的字典序最小(要求不能打乱其他字符的相对位置)。 示例 1: …

Vue 3 30天精进之旅:Day 08 - 组件通信

在Vue 3的开发过程中,组件之间的通信是一个至关重要的概念。理解如何在父子组件、兄弟组件以及通过全局事件总线进行通信,将帮助我们构建更为灵活和可维护的应用。在今天的学习中,我们将探讨以下几个方面: 父子组件之间的通信兄弟…