golang如何优雅的编写事务代码示例

 更新时间:2020年05月26日 14:28:06   作者:雪山飞猪  
这篇文章主要介绍了golang如何优雅的编写事务代码示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

新手程序员大概有如下特点

  • if嵌套经常超过3层、经常出现重复代码、单个函数代码特别长。
  • 只会crud,对语言特性和语言的边界不了解。
  • 不懂面向对象原则和设计模式,以为copy代码就算学会了,常见的是代码职责不明确或者写出万能类
  • 不知道数据结构和算法的重要性,以为靠硬件能解决运行慢的问题
  • 架构不懂,搭建框架不会,搭建环境不会,使用的软件底层原理一问三不知

其实吧,很多人干了很多年,看似是老手,平时工作看似很忙,其实做的都是最简单的活。
这就像去锻炼,有的人每天练的很积极,准时打卡,频繁发朋友圈,貌似是正能量,结果是几年下来体型还是那样,该减的肥肉没少,要增的肌肉没加,为什么会这样?因为从来都是挑最简单最轻松的练

貌似吐槽多了,下面演示一下如何将一坨烂事务代码重构得优雅

需求

执行一个事务,需要调用one、two、three、four、five几个方法,任意一个方法失败,都回滚事务
下面是这些方法的简单模拟,我们用尽可能少的代码模拟一个操作

//开启事务
func beginTransaction() {
 fmt.Println("beginTransaction")
}

//回滚事务
func rollback() {
 fmt.Println("rollback")
}

//提交事务
func commit() {
 fmt.Println("commit")
}

//执行one操作
func one() (err error) {
 fmt.Println("one ok")
 return nil
}

//执行two操作
func two() (err error) {
 fmt.Println("two ok")
 return nil
}

//执行three操作
func three() (err error) {
 fmt.Println("two ok")
 return nil
}

//执行four操作
func four() (err error) {
 fmt.Println("four ok")
 return nil
}

//执行five操作
func five() (err error) {
 err = errors.New("five panic")
 panic("five")
 return err
}

烂代码示例

下面演示开启一个事务,依次执行one、two、three、four、five 5个操作,前四个成功,第五个失败

这是新手程序员常见使用事务的代码风格,其实也不光是事务,所有的代码都可能长下边这样

func demo() (err error) {
 beginTransaction()
 defer func() {
 if e := recover(); e != nil {
  err = fmt.Errorf("%v", e)
  fmt.Printf("%v panic\n", e)
  rollback()
 }
 }()
 if err = one(); err == nil {
 if err = two(); err == nil {
  if err = three(); err == nil {
  if err = four(); err == nil {
   if err = five(); err == nil {
   commit()
   return nil
   } else {
   rollback()
   return err
   }
  } else {
   rollback()
   return err
  }
  } else {
  rollback()
  return err
  }
 } else {
  rollback()
  return err
 }
 } else {
 rollback()
 return err
 }
}

重构套路

一、提前return去除if嵌套

通过提前返回error,来去掉一些else代码,减少嵌套,如下

代码

func demo() (err error) {
 beginTransaction()
 defer func() {
 if e := recover(); e != nil {
  err = fmt.Errorf("%v", e)
  fmt.Printf("%v panic\n", e)
  rollback()
 }
 }()
 if err = one(); err != nil {
 rollback()
 return err
 }
 if err = two(); err != nil {
 rollback()
 return err
 }
 if err = three(); err != nil {
 rollback()
 return err
 }

 if err = four(); err != nil {
 rollback()
 return err
 }
 if err = five(); err != nil {
 rollback()
 return err
 }
 commit()
 return nil
}

先解决嵌套

二、goto+label提取重复代码


代码

func demo() (err error) {
 beginTransaction()
 defer func() {
 if e := recover(); e != nil {
  err = fmt.Errorf("%v", e)
  fmt.Printf("%v panic\n", e)
  rollback()
 }
 }()
 if err = one(); err != nil {
 goto ROLLBACK
 }
 if err = two(); err != nil {
 goto ROLLBACK
 }
 if err = three(); err != nil {
 goto ROLLBACK
 }
 if err = four(); err != nil {
 goto ROLLBACK
 }
 if err = five(); err != nil {
 goto ROLLBACK
 }
 commit()
 return nil
ROLLBACK:
 rollback()
 return err
}

三、封装try-catch统一捕获panic

上面的代码其实还有一点问题

  • defer里有rollback的代码
  • goto虽然好,但是不建议使用

我们可以对panic和defer进行封装,模拟一下try-catch,实现如下



可以看到,rollback只调用了一次,完美的进行了事务代码重构

try-catch.go代码

package exception

type Block struct {
 Try func()
 Catch func(interface{})
 Finally func()
}

func (t Block) Do() {
 if t.Finally != nil {
 defer t.Finally()
 }
 if t.Catch != nil {
 defer func() {
  if r := recover(); r != nil {
  t.Catch(r)
  }
 }()
 }
 t.Try()
}

使用代码

	exception.Block{
		Try: func() {
			beginTransaction()
			if err = one(); err != nil {
				panic(err)
			}
			if err = two(); err != nil {
				panic(err)
			}
			if err = three(); err != nil {
				panic(err)
			}
			if err = four(); err != nil {
				panic(err)
			}
			if err = five(); err != nil {
				panic(err)
			}
			err = nil
			commit()
		},
		Catch: func(e interface{}) {
			rollback()
			fmt.Printf("%v panic\n", e)
			err = fmt.Errorf("%v", e)
		},
	}.Do()
	return err
}

这样,我们就可以用非常少的代码实现事务,并且简单清晰好维护,以上为chenqionghe原创,light weight baby

到此这篇关于golang如何优雅的编写事务代码示例的文章就介绍到这了,更多相关golang 编写事务内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • golang简单tls协议用法完整示例

    golang简单tls协议用法完整示例

    这篇文章主要介绍了golang简单tls用法,分析了tls协议的使用步骤及客户端与服务器端的相关实现代码,需要的朋友可以参考下
    2016-07-07
  • Golang并发编程中Context包的使用与并发控制

    Golang并发编程中Context包的使用与并发控制

    Golang的context包提供了在并发编程中传递取消信号、超时控制和元数据的功能,本文就来介绍一下Golang并发编程中Context包的使用与并发控制,感兴趣的可以了解一下
    2024-11-11
  • go语言中[]*int和*[]int的具体使用

    go语言中[]*int和*[]int的具体使用

    本文主要介绍了go语言中[]*int和*[]int的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • Golang与其他语言不同的九个特性

    Golang与其他语言不同的九个特性

    近来关于对Golang的讨论有很多,七牛的几个大牛们也断定Go语言在未来将会快速发展,并且很可能会取代Java成为互联网时代最受欢迎的编程语言。本文将带你了解它不同于其他语言的九个特性
    2021-09-09
  • 谈谈golang的netpoll原理解析

    谈谈golang的netpoll原理解析

    本文详细介绍了Go语言中netpoll部分的实现细节和协程阻塞调度原理,特别是epoll在Linux环境下的工作原理,Go语言通过将epoll操作放在runtime包中,结合运行时调度功能,实现了高效的协程I/O操作,感兴趣的朋友跟随小编一起看看吧
    2024-11-11
  • Go语言基础go build命令用法及示例详解

    Go语言基础go build命令用法及示例详解

    这篇文章主要为大家介绍了Go语言基础go build命令用法及示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2021-11-11
  • Go语言判断文件或文件夹是否存在的方法

    Go语言判断文件或文件夹是否存在的方法

    这篇文章主要介绍了Go语言判断文件或文件夹是否存在的方法,结合具体实例形式对比分析了Go语言针对文件与目录判断的操作技巧与相关注意事项,需要的朋友可以参考下
    2017-05-05
  • Go 1.13中errors包的新变化示例解析

    Go 1.13中errors包的新变化示例解析

    这篇文章主要为大家介绍了Go 1.13中errors包的新变化示例解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • 详解Golang中Context的原理和使用技巧

    详解Golang中Context的原理和使用技巧

    Golang 的 Context 包,中文可以称之为“上下文”,是用来在 goroutine 协程之间进行上下文信息传递的,这些上下文信息包括 kv 数据、取消信号、超时时间、截止时间等。本文主要介绍了Context的原理和使用技巧,希望对大家有所帮助
    2022-11-11
  • 详解Go 1.22 for循环的两处重要更新

    详解Go 1.22 for循环的两处重要更新

    这篇文章主要详细介绍了Go 1.22 for循环的两处重要更新,Go 1.22 版本于 2024 年 2 月 6 日发布,引入了几个重要的特性和改进,在语言层面上,这个版本对 for 循环进行了两处更新,本文将会对 for 循环的两个更新进行介绍,需要的朋友可以参考下
    2024-02-02

最新评论