func Register() error {
var (
uid int64
err error
)
tx, err := g.DB().Begin()
if err != nil {
return err
}
// 方法退出时检验返回值,
// 如果结果成功则执行tx.Commit()提交,
// 否则执行tx.Rollback()回滚操作。
defer func() {
if err != nil {
tx.Rollback()
} else {
tx.Commit()
}
}()
// 写入用户基础数据
uid, err = AddUserInfo(tx, g.Map{
"name": "john",
"score": 100,
//...
})
if err != nil {
return err
}
// 写入用户详情数据,需要用到上一次写入得到的用户uid
err = AddUserDetail(tx, g.Map{
"uid": uid,
"phone": "18010576259",
//...
})
return err
}
func AddUserInfo(tx *gdb.TX, data g.Map) (int64, error) {
result, err := g.Table("user").TX(tx).Data(data).Insert()
if err != nil {
return 0, err
}
uid, err := result.LastInsertId()
if err != nil {
return 0, err
}
return uid, nil
}
func AddUserDetail(tx *gdb.TX, data g.Map) error {
_, err := g.Table("user_detail").TX(tx).Data(data).Insert()
return err
}
14 Comments
kim
orm中如果对两个表进行了操作,每次操作都是单独声明一个model出来的,那么就是两个独立的事务,如何让后一个事务加入前一个事务中进行统一管理事务呢?
郭强
你好,更新了一下文档,麻烦再看看呢。
kim
通过显式的tx对象来调用确实可以自己手动控制两张表的事务一致性。然而实际开发中代码需要松耦合,可能很多表的操作都是各自独立一个接口内部完成,而且这样写声明式事务的方式对开发人员来说要求比较高,事务一致性、回滚、提交都要开发人员一手一脚来顾及,容易踩坑,建议最好是能够通过一些配置或者方法参数的传递能够在底层智能地处理完事务问题。
郭强
那你需要的就是
Transaction
方法,具体看示例1。kim
实例1不需要手动提交和回滚,会不会导致事务挂起?
郭强
当给定的闭包方法返回的
error
为nil
时,那么闭包执行结束后当前事务自动执行Commit
提交操作;否则自动执行Rollback
回滚操作。kim
明白,谢谢
王一飞
您好,我遇到了一个问题。
当我的`ctxStrict`值设置为true的时候,不管是使用g.DB().Ctx(ctx).Transaction(ctx, func)还是dao.User.Transaction(ctx, func)都会报错
context is required for database operation, did you missing call function Ctx
郭强
v2
新版本已经去掉ctxStrict
功能,后续依靠工程化规范和开发工具去保证,不在组件中打桩判断。白夜
郭强 transaction 闭包方法看起来返回值只能是error,我这边的逻辑是期望类似于 user save之后返回user entity数据,代码如下:
transaction 闭包方法看起来返回值只能是error,而不能返回user的数据。我理解这里的事务除了具有多个操作原子化目的之外,应该还有并发安全的考虑吧。我改成了TX 链式操作,如下:
想问下:TX 链式操作的话,原子化应该依然有,但是并发安全还能保证么?
郭强
在闭包里面你可以设置方法的返回值哈。
如果你实在想用
Begin/Commit/Rollback
操作,链式操作有个链式安全的介绍,如果你使用的是DAO
,那么无论如何都是并发安全的。白夜
郭强
我的目的是想使用 transaction 闭包,并且可以返回除了error之外的数据,比如返回user.Uuid。
但是看 方法定义,是只能返回err ?
我用了dao: dao.User.Ctx(ctx).Data(user).OmitEmpty().Save(). db 操作是并发安全的,这个没有问题,我担心的是我的多个操作是否可以一起 并发安全?
还是说需要我增加 锁 来控制?
潘培钊
user 定义到闭包外面是否可行呢
白夜
赞! 可行。 陷入误区了~