本文中作者展示了golang事务的三种写法。
第一种写法
这种写法非常朴实,程序流程也非常明确,但是事务处理与程序流程嵌入太深,容易遗漏,造成严重的问题
funcDoSomething()(errerror){
tx,err:=db.Begin()
iferr!=nil{
return
}
deferfunc(){
ifp:=recover();p!=nil{
tx.Rollback()
panic(p)//re-throwpanicafterRollback
}
}()
if_,err=tx.Exec(...);err!=nil{
tx.Rollback()
return
}
if_,err=tx.Exec(...);err!=nil{
tx.Rollback()
return
}
//...
err=tx.Commit()
return
}
第二种写法
下面这种写法把事务处理从程序流程抽离了出来,不容易遗漏,但是作用域是整个函数,程序流程不是很清晰
funcDoSomething()(errerror){
tx,err:=db.Begin()
iferr!=nil{
return
}
deferfunc(){
ifp:=recover();p!=nil{
tx.Rollback()
panic(p)//re-throwpanicafterRollback
}elseiferr!=nil{
tx.Rollback()
}else{
err=tx.Commit()
}
}()
if_,err=tx.Exec(...);err!=nil{
return
}
if_,err=tx.Exec(...);err!=nil{
return
}
//...
return
}
第三种写法
写法三是对写法二的进一步封装,写法高级一点,缺点同上
funcTransact(db*sql.DB,txFuncfunc(*sql.Tx)error)(errerror){
tx,err:=db.Begin()
iferr!=nil{
return
}
deferfunc(){
ifp:=recover();p!=nil{
tx.Rollback()
panic(p)//re-throwpanicafterRollback
}elseiferr!=nil{
tx.Rollback()
}else{
err=tx.Commit()
}
}()
err=txFunc(tx)
returnerr
}
funcDoSomething()error{
returnTransact(db,func(tx*sql.Tx)error{
if_,err:=tx.Exec(...);err!=nil{
returnerr
}
if_,err:=tx.Exec(...);err!=nil{
returnerr
}
})
}
我的写法
经过总结和实验,我采用了下面这种写法,defer tx.Rollback() 使得事务回滚始终得到执行。当 tx.Commit() 执行后,tx.Rollback() 起到关闭事务的作用, 当程序因为某个错误中止,tx.Rollback() 起到回滚事务,同事关闭事务的作用。
普通场景
funcDoSomething()(errerror){
tx,_:=db.Begin()
defertx.Rollback()
if_,err=tx.Exec(...);err!=nil{
return
}
if_,err=tx.Exec(...);err!=nil{
return
}
//...
err=tx.Commit()
return
}
循环场景
(1) 小事务 每次循环提交一次 在循环内部使用这种写法的时候,defer 不能使用,所以要把事务部分抽离到独立的函数当中
funcDoSomething()(errerror){
tx,_:=db.Begin()
defertx.Rollback()
if_,err=tx.Exec(...);err!=nil{
return
}
if_,err=tx.Exec(...);err!=nil{
return
}
//...
err=tx.Commit()
return
}
for{
iferr:=DoSomething();err!=nil{
//...
}
}
(2) 大事务 批量提交 大事务的场景和普通场景是一样的,没有任何区别
funcDoSomething()(errerror){
tx,_:=db.Begin()
defertx.Rollback()
for{
if_,err=tx.Exec(...);err!=nil{
return
}
if_,err=tx.Exec(...);err!=nil{
return
}
//...
}
err=tx.Commit()
return
}
-
封装
+关注
关注
126文章
7891浏览量
142927 -
程序
+关注
关注
117文章
3786浏览量
81022 -
函数
+关注
关注
3文章
4329浏览量
62587
原文标题:Golang transaction 事务使用的正确姿势
文章出处:【微信号:magedu-Linux,微信公众号:马哥Linux运维】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
评论