在学习golang中,channel真的是让人捉摸不透的东西,本来我自以为我理解了协程阻塞的用法了,结果就下面这个小例子,我还是在打印输出后才搞明白到底怎么回事?

当然了,这也是我自身对协程这块不太熟造成的呀,另外,学习还真不能想当然,尤其是编程这块,真是要多实践,有时候你不经意的一点小举动,可能都会让你学到东西,甚至让你受益非浅的。

下面就是那个小例子,通过输出调试看结果后,也让我比以前更了解这个channel阻塞模式了。

package main
import (
    "fmt"
    "sync"
)
var wg sync.WaitGroup
var printChar chan int
func prinNums() {
    defer wg.Done()
    for i:=0;i < 2; i++ {
        printChar <- 1111
        fmt.Println(<-printChar)
    }
}

func printChars(){
    defer wg.Done()
    for i:=0;i < 2; i++ {
        fmt.Println("阻1")
        fmt.Println(<-printChar)
        fmt.Println("阻2")
        fmt.Println("出来1")
        printChar <- 1222
        fmt.Println("出来2")
    }
}
func main(){
    printChar = make(chan int)
    
    wg.Add(2)
    
    go prinNums()
    go printChars()

    wg.Wait()
}


输出结果为:
阻1
1111
阻2
出来1
1222
出来2
阻1
1111
阻2
出来1
1222
出来2

看完结果,再仔细分析下就基本明白了。

其实,我最开始是在channel赋值和输出这里犯糊涂了,一直没理解到,比如printNums函数这里

printChar <- 1111
fmt.Println(<-printChar)

我开始想当然的以为直接赋值后,然后就能直接取值了。这就是没理解协程阻塞和调度。当执行到 printChar <- 1111 这里时,因为是阻塞模式,那么它必须等待其它协程将它里面的数据取走,它后面的代码也不会执行,就类似于php的yield,它将按制权让出来给其它的协程。

其它协程将channel里面的值取出来后,继续执行协程阻塞前的那些代码,对比以上的输出就很容易看出来了。

当然了,如果是懂的人,我这个会让人笑掉大牙,但是没关系,这也是我自已自学习路上学到的东西,这就够了。