卓越飞翔博客卓越飞翔博客

卓越飞翔 - 您值得收藏的技术分享站
技术文章34556本站已运行393

为什么通道不采用最后声明的值?

为什么通道不采用最后声明的值?

在PHP中,通道(channel)是一种用于并发编程的重要概念。它允许不同的协程(goroutine)之间进行通信和同步操作。然而,有一个常见的疑问是:“为什么通道不采用最后声明的值?”这个问题涉及到通道的设计原则和使用方式。通道在接收操作时,会阻塞等待发送方发送数据。当发送方发送完数据后,接收方才能继续执行。因此,通道的值是在发送方发送数据时确定的,而不是在接收方接收数据时确定的。这是为了保证通信的可靠性和一致性。所以,即使最后声明的值在接收方接收时已经改变,通道仍然会采用发送时的值。这样做可以避免数据的不一致性和混乱,确保通信的可靠性和准确性。

问题内容

我正在尝试了解频道。在这段代码中,我声明了从 1 到 10 的值。最终声明的值为 10,但是当我打印它时,它总是返回 before(8) 之前两个声明的值。如果有人能解释一下,我会很高兴。

func main() {
    channel := make(chan int, 3)
    isOver := make(chan bool)

    go func() {
        for val := range channel {
            fmt.Println(val)
        }
        isOver <- true
    }()

    channel <- 1
    channel <- 2
    channel <- 3
    channel <- 4
    channel <- 5
    channel <- 6
    channel <- 7
    channel <- 8
    channel <- 9
    channel <- 10
    close(channel)

    fmt.Println("Channel Value is: ", <-channel)

    <-isOver

}

当我将声明更改为 8 时,它返回之前两个声明的值,即 6。

解决方法

因为你为通道设置了3个缓冲区大小!

注意:您的示例在不同的操作系统中具有不同的输出(例如我得到 Channel 值为:0

设置时间。在代码中休眠看看会发生什么。在您的代码中:

func main() {
    channel := make(chan int)
    isOver := make(chan bool)

    go func() {
        for val := range channel {
            fmt.Println(val)
            // sleep 1 second
            time.Sleep(1 * time.Second)
        }
        isOver <- true
    }()

    channel <- 1
    channel <- 2
    channel <- 3
    channel <- 4
    channel <- 5
    channel <- 6
    channel <- 7
    channel <- 8
    channel <- 9
    channel <- 10
    close(channel)

    fmt.Println("Channel Value is: ", <-channel)
    <-isOver
}

输出为:

1
2
3
4
5
6
7
Channel Value is:  8
9
10

注意:了解代码中发生的情况的更好方法是逐步跟踪代码。

发生了什么? 看到这个痕迹:

1 // print 1 and sleep 1s
2,3,4 // stop for get values

3,4 // print 2 and sleep 1
3,4,5 // stop for get value

4,5 // print 3 and sleep 1
4,5,6 // stop for get value

5,6 // print 4 and sleep 1
5,6,7 // stop for get value

6,7 // print 5 and sleep 1
6,7,8 // stop for get value

7,8 // print 6 and sleep 1
7,8,9 // stop for get value

8,9 // print 7 and sleep 1
8,9,10 // stop for get value
// close channel
// in this line get value : fmt.Println("Channel Value is: ", <-channel)

9,10 // before get value
10 // get 9 and sleep 1
// get 10 and sleep 1

// isdone send signal true

在跟踪代码中,当通道获取树值 8,9,10 时,通道关闭,在第 2 步中发生了事情:

1 - 如果 goroutine 快速获取所有值(通道值是:0)
2 - 如果 goroutine 繁忙(通道值为:7 或 8 或 9 或 10)

有关详细信息,请参阅此问题:何时使用缓冲通道 并查看此网站以更好地理解:链接

卓越飞翔博客
上一篇: Go Lang 中继承的结构体数组
下一篇: 返回列表
留言与评论(共有 0 条评论)
   
验证码:
隐藏边栏