這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
1.defer panic recover
defer : golang的defer優雅又簡潔, 是golang的亮點之一。defer在聲明時不會立即執行,而是在函數return後,再按照先進後出的原則依次執行每個defer,一般用於釋放資源、清理資料、記錄日誌、異常處理等。
defer定義的內容必須在return之間就寫上,否則就不會defer了
f, err := os.Open("file")
defer f.Close()
if err != nil { //判斷是否出錯用 err != nil 來判斷
return
}
b, err := ioutil.ReadAll(f)
println(string(b))
defer的智行順序:結果是one,two,three從下向上一次執行
func deferTest(number int) int {
defer func() {number++fmt.Println("three:", number)}()defer func() {number++fmt.Println("two:", number)}()defer func() {number++fmt.Println("one:", number)}()return number}
2.go async.WaitGroup,它能夠一直等到所有的goroutine執行完成,並且阻塞主線程的執行,直到所有的goroutine執行完成。
WaitGroup總共有三個方法:Add(delta int),Done(),Wait()。
Add:添加或者減少等待goroutine的數量
Done:相當於Add(-1)
Wait:執行阻塞,直到所有的WaitGroup數量變成0
var waitGroup sync.WaitGroup //定義一個同步等待的組
也可以這樣產生:
waitGroup
:= &sync.WaitGroup{}
waitGroup.Add(1) //添加一個計數
go function() {} //執行任務
waitGroup.Wait() //阻塞直到所有任務完成
package mainimport ("fmt""sync""time")func main() {var wg sync.WaitGroupfor i := 0; i < 5; i = i + 1 {wg.Add(1)go func(n int) {// defer wg.Done()defer wg.Add(-1)EchoNumber(n)}(i)}wg.Wait()}func EchoNumber(i int) {time.Sleep(3e9)fmt.Println(i)}
程式很簡單,只是將每次迴圈的數量過3秒鐘輸出。那麼,這個程式如果不用WaitGroup,那麼將看不見輸出結果。因為goroutine還沒執行完,主線程已經執行完畢。注釋的defer wg.Done()和defer wg.Add(-1)作用一樣。這個很好,原來執行指令碼,都是使用time.Sleep,用一個估計的時間等到子線程執行完。WaitGroup很好。雖然chanel也能實現,但是覺得如果涉及不到子線程與主線程資料同步,這個感覺不錯。
備忘:https://studygolang.com/articles/319