Categories: GoFrame 教程

GoFrame 链式操作-事务处理

Model​对象也可以通过​TX​事务对象创建,通过事务对象创建的​Model​对象与通过​DB​数据库对象创建的​Model​对象功能是一样的,只不过前者的所有操作都是基于事务,而当事务提交或者回滚后,对应的​Model​对象不能被继续使用,否则会返回错误。因为该​TX​对象不能被继续使用,一个事务对象仅对应于一个事务流程,​Commit/Rollback​后即结束。

示例1,通过Transaction

为方便事务操作,​gdb​提供了事务的闭包操作,通过​Transaction​方法实现,该方法定义如下:

func (db DB) Transaction(ctx context.Context, f func(ctx context.Context, tx *TX) error) (err error)

当给定的闭包方法返回的​error​为​nil​时,那么闭包执行结束后当前事务自动执行​Commit​提交操作;否则自动执行​Rollback​回滚操作。

如果闭包内部操作产生​panic​中断,该事务也将进行回滚。

func Register() error {
 return db.Transaction(ctx, func(ctx context.Context, tx *gdb.TX) error {
  var (
   result sql.Result
   err    error
  )
  // 写入用户基础数据
  result, err = tx.Table("user").Insert(g.Map{
   "name":  "john",
   "score": 100,
   //...
  })
  if err != nil {
   return err
  }
  // 写入用户详情数据,需要用到上一次写入得到的用户uid
  result, err = tx.Table("user_detail").Insert(g.Map{
   "uid":   result.LastInsertId(),
   "phone": "18010576258",
   //...
  })
  return err
 })
}

示例2,通过TX链式操作

我们也可以在链式操作中通过​TX​方法切换绑定的事务对象。多次链式操作可以绑定同一个事务对象,在该事务对象中执行对应的链式操作。

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
}

admin

这个人很懒,什么都没有留下~

Share
Published by
admin

Recent Posts

聊聊vue3中的defineProps

在Vue 3中,defineP…

1 周 ago

在 Chrome 中删除、允许和管理 Cookie

您可以选择删除现有 Cooki…

2 周 ago

自定义指令:聊聊vue中的自定义指令应用法则

今天我们来聊聊vue中的自定义…

3 周 ago

聊聊Vue中@click.stop和@click.prevent

一起来学下聊聊Vue中@cli…

4 周 ago

Nginx 基本操作:启动、停止、重启命令。

我们来学习Nginx基础操作:…

1 月 ago

Vue3:手动清理keep-alive组件缓存的方法

Vue3中手动清理keep-a…

1 月 ago