Golang GORM系列:GORM事务及错误处理
在数据库管理领域,确保数据完整性至关重要。GORM是健壮的Go对象关系映射库,它为开发人员提供了维护数据一致性和优雅地处理错误的基本工具。本文是掌握GORM事务和错误处理的全面指南。我们将深入研究如何使用事务来保证原子性,并探索有效处理错误和回滚的策略,以保持Go项目中数据库操作的可靠性和健壮性。

在GORM中处理事务
关重要的作用。GORM的事务支持使您能够将多个数据库操作作为单个工作单元执行。
步骤1:开始一个事务
使用GORM的‘ Begin ’方法启动一个事务:
tx := db.Begin()
步骤2:执行操作
在事务中执行数据库操作:
if err := tx.Create(&User{Name: "Alice"}).Error; err != nil {tx.Rollback()return err
}
步骤3:提交或回滚
执行操作后,选择提交或回滚事务:
if err := tx.Commit().Error; err != nil {tx.Rollback()return err
}
处理GORM中的错误和回滚
优雅的错误处理和回滚对于在发生故障时保持数据一致性和完整性至关重要。
步骤1:处理错误
检查错误并适当处理它们:
if err := tx.Create(&User{Name: "Bob"}).Error; err != nil {// Handle error
}
步骤2:执行回滚
如果出现错误,请回滚事务以确保数据完整性:
if err := tx.Commit().Error; err != nil {tx.Rollback()return err
}
GORM中的嵌套事务
GORM支持嵌套事务,允许你将特定的操作封装在它们自己的事务边界内。
outer := db.Begin()// Perform operations in the outer transactioninner := outer.Begin()// Perform operations in the inner transactionif err := inner.Commit().Error; err != nil {inner.Rollback()outer.Rollback()return err
}if err := outer.Commit().Error; err != nil {outer.Rollback()return err
}
完整示例
假设我们有两个模型 User 和 Order,当创建一个用户时,同时需要为该用户创建一个订单。如果在创建订单时发生错误,我们希望回滚整个操作,包括用户的创建。
分步说明
- 定义模型
首先定义User和Order模型。
type User struct {ID uintName stringEmail string
}type Order struct {ID uintUserID uintProduct string
}
开启外层事务
使用 db.Begin() 开启一个外层事务。
tx := db.Begin()
if tx.Error != nil {log.Fatalf("Failed to begin transaction: %v", tx.Error)
}
创建用户
在外层事务中创建用户。
user := User{Name: "John Doe", Email: "john@example.com"}
if err := tx.Create(&user).Error; err != nil {tx.Rollback() // 回滚外层事务log.Fatalf("Failed to create user: %v", err)
}
开启内层事务
在外层事务中开启一个内层事务。
nestedTx := tx.Begin()
if nestedTx.Error != nil {tx.Rollback() // 回滚外层事务log.Fatalf("Failed to begin nested transaction: %v", nestedTx.Error)
}
创建订单
在内层事务中创建订单。
order := Order{UserID: user.ID, Product: "Laptop"}
if err := nestedTx.Create(&order).Error; err != nil {nestedTx.Rollback() // 回滚内层事务tx.Rollback() // 回滚外层事务log.Fatalf("Failed to create order: %v", err)
}
提交内层事务
如果内层事务成功,提交内层事务。
if err := nestedTx.Commit().Error; err != nil {tx.Rollback() // 回滚外层事务log.Fatalf("Failed to commit nested transaction: %v", err)
}
提交外层事务
如果外层事务成功,提交外层事务。
if err := tx.Commit().Error; err != nil {log.Fatalf("Failed to commit transaction: %v", err)
}
完整代码
package mainimport ("log""gorm.io/driver/sqlite""gorm.io/gorm"
)type User struct {ID uintName stringEmail string
}type Order struct {ID uintUserID uintProduct string
}func main() {// 连接数据库db, err := gorm.Open(sqlite.Open("test.db"), &gorm.Config{})if err != nil {log.Fatalf("Failed to connect to database: %v", err)}// 自动迁移模型db.AutoMigrate(&User{}, &Order{})// 开启外层事务tx := db.Begin()if tx.Error != nil {log.Fatalf("Failed to begin transaction: %v", tx.Error)}// 创建用户user := User{Name: "John Doe", Email: "john@example.com"}if err := tx.Create(&user).Error; err != nil {tx.Rollback() // 回滚外层事务log.Fatalf("Failed to create user: %v", err)}// 开启内层事务nestedTx := tx.Begin()if nestedTx.Error != nil {tx.Rollback() // 回滚外层事务log.Fatalf("Failed to begin nested transaction: %v", nestedTx.Error)}// 创建订单order := Order{UserID: user.ID, Product: "Laptop"}if err := nestedTx.Create(&order).Error; err != nil {nestedTx.Rollback() // 回滚内层事务tx.Rollback() // 回滚外层事务log.Fatalf("Failed to create order: %v", err)}// 提交内层事务if err := nestedTx.Commit().Error; err != nil {tx.Rollback() // 回滚外层事务log.Fatalf("Failed to commit nested transaction: %v", err)}// 提交外层事务if err := tx.Commit().Error; err != nil {log.Fatalf("Failed to commit transaction: %v", err)}log.Println("User and order created successfully")
}
补充解释
- 外层事务:
tx是外层事务,负责创建用户。 - 内层事务:
nestedTx是内层事务,负责创建订单。 - 回滚机制:如果内层事务失败,外层事务也会回滚,确保数据一致性。
- 提交顺序:先提交内层事务,再提交外层事务。
通过这种方式,你可以在 GORM 中实现嵌套事务,确保多个操作的原子性。
最后总结
事务和错误处理是可靠数据库操作的基础。借助GORM强大的事务支持和错误处理技术,你可以确保Go应用程序中的数据完整性并保持一致的状态。通过掌握处理事务、优雅地处理错误和理解回滚的重要性的艺术,你已经获得了驾驭复杂场景和自信地处理故障的技能。
相关文章:
Golang GORM系列:GORM事务及错误处理
在数据库管理领域,确保数据完整性至关重要。GORM是健壮的Go对象关系映射库,它为开发人员提供了维护数据一致性和优雅地处理错误的基本工具。本文是掌握GORM事务和错误处理的全面指南。我们将深入研究如何使用事务来保证原子性,并探索有效处理…...
如何实现对 ELK 各组件的监控?试试 Metricbea
上一章基于 Filebeat 的日志收集使用Filebeat收集文件中的日志,而Metricbeat则是收集服务器存活性监测和系统指标的指标。 1. Filebeat和Metricbeat的区别 特性FilebeatHeartbeat作用收集和转发日志监测服务可用性数据来源服务器上的日志文件远程主机、API、服务主…...
图像处理篇---基本Python图像处理
文章目录 前言1. 图像灰度化原理Python 实现 2. 图像二值化原理:Python 实现 3. 图像掩膜(Mask)原理Python 实现 4. 腐蚀(Erosion)和膨胀(Dilation)原理Python 实现 5. 其他常用图像操作(1) 图像…...
使用PHP爬虫获取1688商品分类:实战案例指南
在电商领域,商品分类信息是商家进行市场调研、选品分析和竞争情报收集的重要基础。1688作为国内领先的B2B电商平台,提供了丰富且详细的商品分类数据。通过PHP爬虫技术,我们可以高效地获取这些分类信息,为商业决策提供有力支持。 …...
【微服务学习二】nacos服务发现与负载均衡
nacos服务发现 想要开启服务发现,需要在main函数上添加 EnableDiscoveryClient 注解 然后我们编写一个controller类来查询nacos中注册的所有微服务以及对应的ip端口号 Controller public class DiscoveryController {AutowiredDiscoveryClient discoveryClient;//…...
深入剖析推理模型:从DeepSeek R1看LLM推理能力构建与优化
著名 AI 研究者和博主 Sebastian Raschka 又更新博客了。原文地址:https://sebastianraschka.com/blog/2025/understanding-reasoning-llms.html。这一次,他将立足于 DeepSeek 技术报告,介绍用于构建推理模型的四种主要方法,也就是…...
有哪些滤波,原理是什么,分别在什么时候用
均值滤波(Average Filtering) 原理:通过计算像素点邻域内像素值的平均值来作为该像素点滤波后的新值。例如,对于一个 3x3 的邻域,将 9 个像素值相加然后除以 9 得到滤波后的像素值。优点:简单易实现&#x…...
小初高各学科教材,PDF电子版下载
链接:https://pan.quark.cn/s/7c2125f648e2 小初高中电子课本资料pdf合集 高中各科教材 (部分举例) - 语文:新人教版、旧人教版、苏教版等 - 数学:人教A版、沪教版、鄂教版等 - 英语:重大版、人教版…...
Kafka分区管理大师指南:扩容、均衡、迁移与限流全解析
#作者:孙德新 文章目录 分区分配操作(kafka-reassign-partitions.sh)1.1 分区扩容、数据均衡、迁移(kafka-reassign-partitions.sh)1.2、修改topic分区partition的副本数(扩缩容副本)1.3、Partition Reassign场景限流1.4、节点内副本移动到不…...
java后端开发day14--之前练习的总结和思考
1.感受 这两天学点儿新的就直接上手打代码,真的是累死个人。我唯一的感受就是,课听完了,代码也跟着打完了(是的,跟着打的,没自己打),感觉自己脑袋里乱乱的,对代码的分区…...
[运输时间]
运输时间 真题目录: 点击去查看 E 卷 200分题型 题目描述 M(1 ≤ M ≤ 20)辆车需要在一条不能超车的单行道到达终点,起点到终点的距离为 N(1 ≤ N ≤ 400)。 速度快的车追上前车后,只能以前车的速度继续行驶,求最后一辆车到达目的地花费的时间。 注:每辆车固定间隔…...
【愚公系列】《Python网络爬虫从入门到精通》008-正则表达式基础
标题详情作者简介愚公搬代码头衔华为云特约编辑,华为云云享专家,华为开发者专家,华为产品云测专家,CSDN博客专家,CSDN商业化专家,阿里云专家博主,阿里云签约作者,腾讯云优秀博主,腾讯云内容共创官,掘金优秀博主,亚马逊技领云博主,51CTO博客专家等。近期荣誉2022年度…...
理解 WebGPU 中的 GPUQueue:GPU 的命令队列
在现代图形编程中,与 GPU 的交互变得越来越高效和灵活,而 WebGPU API 的出现更是为 Web 开发者带来了强大的图形处理能力。其中, GPUQueue 作为 WebGPU 的核心接口之一,扮演着至关重要的角色。本文将详细介绍 GPUQueue 的概…...
AIoT时代来临,物联网技术如何颠覆未来生活?
在这个万物互联的时代,“物联网”(IoT)正以前所未有的速度改变我们的生活,而“AIoT”则是在物联网基础上融入人工智能技术,赋予设备更高的智能和自主决策能力。随着5G、边缘计算和云技术的不断发展,物联网正…...
基于SpringBoot的电影院售票管理系统
作者:计算机学姐 开发技术:SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等,“文末源码”。 专栏推荐:前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏:…...
CanMV的刷新比Openmv强
今天使用CanMV k230的板子,发现CanMV的刷新比Openmv强,速度快,不用再次拍照刷新,写一次就能在ide屏幕上同时显示。 参考一下CanMV K230拍照保存Demo - CanMV(K210 / K230) - 01科技 | 01Studio Takephot(…...
深度学习与人工智能:解锁未来的无限可能
在当今这个科技飞速发展的时代,深度学习和人工智能已不再只是科幻小说中的概念,它们正以惊人的速度渗透到我们生活的方方面面,从智能手机上的语音助手到医疗领域的疾病诊断,从自动驾驶汽车到金融市场的风险预测,其影响…...
广东茂名能源国际会议(IS-ESE 2025)
能源科学与工程国际研讨会(IS-ESE 2025)将于4月18-20日中国茂名召开,聚焦能源科学与工程,录用文章将由IOP出版并提交EI索引,旨在促进国际合作与交流,诚邀海内外专家学者参会。 征稿主题集中但不限于“能源…...
2025年02月14日Github流行趋势
项目名称:data-formulator 项目地址url:https://github.com/microsoft/data-formulator项目语言:TypeScript历史star数:5921今日star数:820项目维护者:Chenglong-MS, danmarshall, apps/dependabot, micros…...
MySQL-SQL
1.客户端内置命令 客户端内置命令客户端独有,可能不同数据库产品的客户端内置命令存在很大差异,不像SQL命令有标准规范。 help \h ? \? 这四个命令都可以输出帮助文档查看客户端内置命令 ?(\?)“帮助”…...
华为云+硅基流动使用Chatbox接入DeepSeek-R1满血版671B
华为云硅基流动使用Chatbox接入DeepSeek-R1满血版671B 硅基流动 1.1 注册登录 1.2 实名认证 1.3 创建API密钥 1.4 客户端工具 OllamaChatboxCherry StudioAnythingLLM 资源包下载: AI聊天本地客户端 接入Chatbox客户端 点击设置 选择SiliconFloW API 粘贴1.3创…...
一文详解机器视觉环形光源,视觉检测中的环形光源应用
环形光源 是机器视觉系统中常用的一种照明设备,通常用于提供均匀、无阴影的照明,特别适合检测物体的表面特征、边缘和轮廓。以下是关于环形光源的详细介绍: 环形光源的结构与特点 结构:环形光源由多个 LED 灯珠均匀排列成环形,通常安装在相机镜头周围。 光源的内径和外径…...
动态建表并插入数据
Service层根据解析到的数据在Mysql数据库中动态建表并插入数据 以Easy Excel解析得到的文件为例 Slf4j Service public class ExcelImportServiceImpl implements ExcelImportService {Autowired private ExcelImportDao dao; Value("${source.url}") private Stri…...
【github】docker realtime
Linux和Docker实时指南,适用于Ubuntu实时内核和PREEMPT_RT ReadMe.md 作者:Tobit Flatscher(2021 - 2024) 概述 本指南解释了如何在Linux操作系统内开发/部署运行实时代码的Docker容器。因此,它会带你了解…...
MySQL官网驱动下载(jar包驱动和ODBC驱动)【详细教程】
1.打开MySQL的官网,选择下载(Download) MySQL[这里是图片001]https://www.mysql.com/cn/ 2.往下划点击MySQL Community(GPL)Downloads 3.要下载MySQL的jar包的选择Connector/J 4.进入后,根据自己的需求选择相应的版本 5.下载完成后,进行解压…...
jenkins服务启动-排错
服务状态为active (exited) 且进程不在 查看/etc/rc.d/init.d/jenkins配置 获取配置参数 [rootfy-jenkins-prod jenkins]# cat /etc/rc.d/init.d/jenkins | grep -v #JENKINS_WAR"/usr/lib/jenkins/jenkins.war" test -r "$JENKINS_WAR" || { echo "…...
JavaScript设计模式 -- 适配器模式
在软件开发中,经常会遇到这样的情况:现有的类或第三方库提供的接口与系统中期望的接口不匹配。如果直接修改已有代码风险较大或者不可行,这时适配器模式(Adapter Pattern)就能派上用场。适配器模式通过创建一个包装类&…...
Redis7.0八种数据结构底层原理
导读 本文介绍redis应用数据结构与物理存储结构,共八种应用数据结构和 一. 内部数据结构 1. sds sds是redis自己设计的字符串结构有以下特点: jemalloc内存管理预分配冗余空间二进制安全(c原生使用\0作为结尾标识,所以无法直接存储\0)动态计数类型(根据字符串长度动态选择…...
细说STM32F407单片机RTC的备份寄存器原理及使用方法
目录 一、备份寄存器的功能 二、示例功能 三、项目设置 1、晶振、DEBUG、CodeGenerator、USART6 2、RTC 3、NVIC 4、GPIO 及KEYLED 四、软件设计 1、main.h 2、main.c 3、rtc.c 4、keyled.c、keyled.h 五、运行调试 本实例旨在介绍备份寄存器的作用。本实例继续使…...
spring 学习 (注解)
目录 前言 常用的注解 须知 1 Conponent注解 demo(案例) 2 ControllerServiceRepository demo(案例) 3 ScopeLazyPostConstructPreDestroy demo(案例) 4 ValueAutowiredQualifierResource demo(案例) 5 Co…...
