go - Go 中如何比较来自 channel 的值

标签 go concurrency channel goroutine

我有两个 channel ,首先给我一些字符串,我需要将它们过滤成相同的值,然后将结果发送到第二个 channel

func main() {
    c := make(chan string, 5)
    o := make(chan string, 5)
    arr := []string{"aa", "ab", "ab", "bb", "bb", "ba", "cc"}
    for _, v := range arr {
        c <- v
        go removeDuplicates(c, o)
        time.Sleep(1 * time.Second)
        fmt.Println("output: ", <-o)
    }
}

func removeDuplicates(cIn, cOut chan string) {
   last := ""
   for cur, isOpen := <-cIn; isOpen; {
      if cur != last {
        fmt.Printf("val: %s, last: %s\n", cur, last) 
        last = cur
        cOut <- cur
        //close(cOut)
      }
   }
}

我尝试将以前的值保存到“last”变量,但是当我运行程序时,“last”为空

val: aa, last: 
output:  aa
val: ab, last: 
output:  ab
val: ab, last:

我也不知道在这种情况下什么时候需要关闭哪些 channel 。 感谢您的帮助和关注

最佳答案

首先修复removeDuplicates()

问题是您的 for statement 中有一个空的 post 语句:

for cur, isOpen := <-cIn; isOpen; {
    // ..
}

因此您从 cIn channel 接收了一次,但您再也没有接收到更多,您在 post 语句中什么都不做,所以您只是无休止地重复循环体。

一旦循环体执行完毕,又要接收:

for cur, isOpen := <-cIn; isOpen; cur, isOpen = <-cIn {
    // ..
}

这样,输出将是(在 Go Playground 上尝试):

val: aa, last: 
output:  aa
val: ab, last: aa
output:  ab
val: ab, last: 
output:  ab
val: bb, last: ab
output:  bb
val: bb, last: 
output:  bb
val: ba, last: ab
output:  ba
val: cc, last: 
output:  cc

但最好是在 channel 上使用 for range:

for cur := range cIn {
    if cur != last {
        fmt.Printf("val: %s, last: %s\n", cur, last)
        last = cur
        cOut <- cur
    }
}

这输出相同。在 Go Playground 上试试这个.

现在开始修复 main()

我们看到“无效”输出,输出中的值仍然重复。

这是因为您启动了多个运行 removeDuplicates() 的 goroutine。这很糟糕,因为在输入 channel 上发送的值将被多个 goroutine 接收,如果重复的值没有被一个 goroutine 接收,它们仍然可以被检测为唯一的,因此相同的值将被多次发送到输出.

让单个生产者在输入 channel 上发送所有值,发送完所有值后,关闭 channel 。

让一个 goroutine 过滤值,使用 for range,一旦循环退出(所有输入值都被消耗),关闭输出 channel 。

并让一个 goroutine 从输出 channel 接收值,使用 for range,这样你就可以消除丑陋的 time.Sleep:

func main() {
    c := make(chan string, 5)
    o := make(chan string, 5)

    go func() {
        arr := []string{"aa", "ab", "ab", "bb", "bb", "ba", "cc"}
        for _, v := range arr {
            c <- v
        }
        close(c)
    }()

    go removeDuplicates(c, o)

    for v := range o {
        fmt.Println("output: ", v)
    }
}

func removeDuplicates(cIn chan string, cOut chan string) {
    last := ""
    for cur := range cIn {
        if cur != last {
            fmt.Printf("val: %s, last: %s\n", cur, last)
            last = cur
            cOut <- cur
        }
    }
    close(cOut)
}

这将输出(在 Go Playground 上尝试):

val: aa, last: 
val: ab, last: aa
val: bb, last: ab
val: ba, last: bb
val: cc, last: ba
output:  aa
output:  ab
output:  bb
output:  ba
output:  cc

关于go - Go 中如何比较来自 channel 的值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63630193/

相关文章:

google-app-engine - Golang 卡在 WaitGroup

unicode - 在 Go 中读取带有 BOM 的文件

java - 如何解决 Future<CAP#1> 无法转换为 Future<Void> 的问题?

go - 重新连接时丢弃消息的可重新连接的 websocket

go - 使用 channel 下单

api - 获取名称中带有空格的 channel 的统计信息

go - 如何从函数返回 slice 的更改值?

go - 是否可以在不使用界面的情况下使其工作?

java - 新线程不显示框架

java - 处理线程池中不同优先级的任务