這篇文章主要講解了golang編寫事務(wù)的方法,內(nèi)容清晰明了,對此有興趣的小伙伴可以學(xué)習(xí)一下,相信大家閱讀完之后會有幫助。
創(chuàng)新互聯(lián)專業(yè)為企業(yè)提供昌黎網(wǎng)站建設(shè)、昌黎做網(wǎng)站、昌黎網(wǎng)站設(shè)計、昌黎網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計與制作、昌黎企業(yè)網(wǎng)站模板建站服務(wù),十多年昌黎做網(wǎng)站經(jīng)驗,不只是建網(wǎng)站,更提供有價值的思路和整體網(wǎng)絡(luò)服務(wù)。
前言
新手程序員大概有如下特點
其實吧,很多人干了很多年,看似是老手,平時工作看似很忙,其實做的都是最簡單的活。
這就像去鍛煉,有的人每天練的很積極,準時打卡,頻繁發(fā)朋友圈,貌似是正能量,結(jié)果是幾年下來體型還是那樣,該減的肥肉沒少,要增的肌肉沒加,為什么會這樣?因為從來都是挑最簡單最輕松的練
貌似吐槽多了,下面演示一下如何將一坨爛事務(wù)代碼重構(gòu)得優(yōu)雅
需求
執(zhí)行一個事務(wù),需要調(diào)用one、two、three、four、five幾個方法,任意一個方法失敗,都回滾事務(wù)
下面是這些方法的簡單模擬,我們用盡可能少的代碼模擬一個操作
//開啟事務(wù) func beginTransaction() { fmt.Println("beginTransaction") } //回滾事務(wù) func rollback() { fmt.Println("rollback") } //提交事務(wù) func commit() { fmt.Println("commit") } //執(zhí)行one操作 func one() (err error) { fmt.Println("one ok") return nil } //執(zhí)行two操作 func two() (err error) { fmt.Println("two ok") return nil } //執(zhí)行three操作 func three() (err error) { fmt.Println("two ok") return nil } //執(zhí)行four操作 func four() (err error) { fmt.Println("four ok") return nil } //執(zhí)行five操作 func five() (err error) { err = errors.New("five panic") panic("five") return err }
爛代碼示例
下面演示開啟一個事務(wù),依次執(zhí)行one、two、three、four、five 5個操作,前四個成功,第五個失敗
這是新手程序員常見使用事務(wù)的代碼風格,其實也不光是事務(wù),所有的代碼都可能長下邊這樣
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 } }
重構(gòu)套路
一、提前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提取重復(fù)代碼
代碼
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統(tǒng)一捕獲panic
上面的代碼其實還有一點問題
我們可以對panic和defer進行封裝,模擬一下try-catch,實現(xiàn)如下
可以看到,rollback只調(diào)用了一次,完美的進行了事務(wù)代碼重構(gòu)
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 }
這樣,我們就可以用非常少的代碼實現(xiàn)事務(wù),并且簡單清晰好維護,以上為chenqionghe原創(chuàng),light weight baby
看完上述內(nèi)容,是不是對golang編寫事務(wù)的方法有進一步的了解,如果還想學(xué)習(xí)更多內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。