golang goroutine順序輸出方式
range字符串,使用goroutine打印
因為goroutine隨機執行
for _, v := range str { go func() { fmt.Println(string(v)) }() }
輸出:
5
5
5
5
5
可以使用chan順序輸出
for _, c := range str{ ch := make(chan rune) go func(ch <-chan rune) { key := <-ch fmt.Println(string(key)) }(ch) ch <- c }
輸出:
1
2
3
4
5
補充:golang goroutine順序循環打印ABC
分別使用sync.WaitGroup和context
使用sync.WaitGroup, 可控制循環次數
package main import ( "fmt" "sync" ) //控制循環次數 var count = 5 func main() { wg := sync.WaitGroup{} chanA := make(chan struct{}, 1) chanB := make(chan struct{}, 1) chanC := make(chan struct{}, 1) chanA <- struct{}{} wg.Add(3) go printA(&wg, chanA, chanB) go printB(&wg, chanB, chanC) go printC(&wg, chanC, chanA) wg.Wait() } func printA(wg *sync.WaitGroup, chanA, chanB chan struct{}) { defer wg.Done() for i := 0; i < count; i++ { <-chanA fmt.Println("A") chanB <- struct{}{} } } func printB(wg *sync.WaitGroup, chanB, chanC chan struct{}) { defer wg.Done() for i := 0; i < count; i++ { <-chanB fmt.Println("B") chanC <- struct{}{} } } func printC(wg *sync.WaitGroup, chanC, chanA chan struct{}) { defer wg.Done() for i := 0; i < count; i++ { <-chanC fmt.Println("C") chanA <- struct{}{} } }
使用context.WithCancel,通過time.Sleep控制打印數量
package main import ( "context" "fmt" "time" ) func main() { chanA := make(chan struct{}, 1) chanB := make(chan struct{}, 1) chanC := make(chan struct{}, 1) chanA <- struct{}{} ctx1, cancel1 := context.WithCancel(context.Background()) ctx2, cancel2 := context.WithCancel(context.Background()) ctx3, cancel3 := context.WithCancel(context.Background()) go printA(ctx1, chanA, chanB) go printB(ctx2, chanB, chanC) go printC(ctx3, chanC, chanA) time.Sleep(100 * time.Microsecond) cancel1() cancel2() cancel3() } func printA(ctx context.Context, chanA, chanB chan struct{}) { for { select { case <-ctx.Done(): fmt.Println("cancel by parent") // 不會輸出 return case <-chanA: fmt.Println("A") chanB <- struct{}{} } } } func printB(ctx context.Context, chanB, chanC chan struct{}) { for { select { case <-ctx.Done(): fmt.Println("cancel by parent") // 不會輸出 return case <-chanB: fmt.Println("B") chanC <- struct{}{} } } } func printC(ctx context.Context, chanC, chanA chan struct{}) { for { select { case <-ctx.Done(): fmt.Println("cancel by parent") // 不會輸出 return case <-chanC: fmt.Println("C") chanA <- struct{}{} } } }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- Golang 標準庫 tips之waitgroup詳解
- 詳解Go語言中Goroutine退出機制的原理及使用
- Go並發編程中sync/errGroup的使用
- Golang優雅保持main函數不退出的辦法
- Go語言如何輕松編寫高效可靠的並發程序