新手程序员大概有以下特色程序员
其实吧,不少人干了不少年,看似是老手,平时工做看似很忙,其实作的都是最简单的活。
这就像去锻炼,有的人天天练的很积极,准时打卡,频繁发朋友圈,貌似是正能量,结果是几年下来体型仍是那样,该减的肥肉没少,要增的肌肉没加,为何会这样?由于历来都是挑最简单最轻松的练算法
貌似吐槽多了,下面演示一下如何将一坨烂事务代码重构得优雅设计模式
执行一个事务,须要调用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 } }
经过提早返回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 }
先解决嵌套设计
代码3d
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 }
上面的代码其实还有一点问题
咱们能够对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