进行频道发布,接收和进行常规调度

问题描述

我有以下一段go脚本,并且对读取数据的通道和go例程的执行顺序有疑问。

formdata.append("image",this.state.image);

输出

package main

import "fmt"

func squares(c chan int) {
    fmt.Println("started squares goroutine...")
    for num := range c {
        //num := <-c
        fmt.Println(num * num)
    }
    fmt.Println("exiting squares goroutine...")
}

func main() {
    fmt.Println("main() started")
    c := make(chan int,3)

    go squares(c)

    c <- 1
    printChannel(c)
    c <- 2
    printChannel(c)
    c <- 3
    printChannel(c)
    c <- 4
    fmt.Println("after publish 4")
    printChannel(c)
    c <- 5
    fmt.Println("after publish 5")
    printChannel(c)
    c <- 6
    printChannel(c)
    c <- 7
    printChannel(c)
    c <- 8
    printChannel(c)

    fmt.Println("main() stopped")
}

func printChannel(c chan int) {
    fmt.Println("length: ",len(c))
    fmt.Println("capacity: ",cap(c))
    
}

问题:

  1. 所创建的通道ch的容量为3,但是在通道4上发布该通道之前,如果容量为3,则在方格例行程序中不读取该通道。

  2. 从通道中读取所有数字后,再次将5号发布到该通道,但是通道的长度仍为0,为什么?

代码的Goplayground链接https://play.golang.org/p/TmCcZt5n58f

解决方法

无法保证执行goroutine的顺序,也不能保证goroutine何时中断(I / O,包括打印到stdout将是重新计划的最常见原因)。

第一个问题:您可以看到16之前正在打印after publish 4。这意味着squares goroutine从通道读取并在主goroutine可以开始打印信息之前打印正方形。到主goroutine尝试打印通道的长度时,4已经消失了。

类似地,在将5发送到通道后,squares goroutine已收到它,但尚未打印出来。打印之前它被打断了。这意味着该通道再次为空,但是我们还没有看到25

如果多次运行示例,将会看到不同的输出。有时squares goroutine一直到64,有时却没有。这是完全正常的。在普通代码中,您需要确保在退出之前等待数据被使用,并且您不必依赖不同goroutine的执行顺序或时间。