登录
首页 >  Golang >  Go问答

如何等待缓冲通道(信号量)为空?

来源:stackoverflow

时间:2024-04-26 10:54:38 298浏览 收藏

今天golang学习网给大家带来了《如何等待缓冲通道(信号量)为空?》,其中涉及到的知识点包括等等,无论你是小白还是老手,都适合看一看哦~有好的建议也欢迎大家在评论留言,若是看完有所收获,也希望大家能多多点赞支持呀!一起加油学习~

问题内容

我有一个整数切片,它们是同时操作的:

ints := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}

我使用缓冲通道作为信号量,以获得同时运行的 go 例程的上限:

sem := make(chan struct{}, 2)

for _, i := range ints {
  // acquire semaphore
  sem <- struct{}{}

  // start long running go routine
  go func(id int, sem chan struct{}) {
    // do something

    // release semaphore
    <- sem
  }(i, sem)
}

上面的代码在到达最后一个或最后两个整数之前工作得很好,因为程序在最后的 go 例程完成之前结束。

问题:如何等待缓冲通道耗尽?


解决方案


您不能以这种方式使用信号量(在本例中为通道)。当您处理值和调度更多 goroutine 时,无法保证它不会为空。在这种情况下,这不是一个问题,因为您是同步分派工作,但因为没有无竞争的方法来检查通道的长度,所以没有基元来等待通道的长度达到 0。

使用 sync.waitgroup 等待所有 goroutine 完成

sem := make(chan struct{}, 2)

var wg sync.waitgroup

for _, i := range ints {
    wg.add(1)
    // acquire semaphore
    sem <- struct{}{}
    // start long running go routine
    go func(id int) {
        defer wg.done()
        // do something
        // release semaphore
        <-sem
    }(i)
}

wg.wait()

使用“工作池”来处理您的数据。它比为 each int 运行 goroutine、为其中的变量分配内存等等更便宜......

ints := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}

ch := make(chan int)

var wg sync.WaitGroup

// run worker pool
for i := 2; i > 0; i-- {
    wg.Add(1)

    go func() {
        defer wg.Done()

        for id := range ch {
            // do something
            fmt.Println(id)
        }
    }()
}

// send ints to workers
for _, i := range ints {
    ch <- i
}

close(ch)

wg.Wait()

理论要掌握,实操不能落!以上关于《如何等待缓冲通道(信号量)为空?》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>