package main
import (
"fmt"
_ "reflect"
"sync"
"time"
)
func writeData(intChan chan int) {
for i := 1; i <= 8000; i++ {
intChan <- i
//fmt.Printf("writeData()写入数据%v", i)
}
close(intChan)
//fmt.Println(intChan)
}
func judgData(intChan chan int, resultChan chan int, exitChan chan bool, done func()) {
defer done()
time.Sleep(time.Millisecond * 1000)
flag := true
//time.Sleep(time.Second *1)
for {
num, ok := <-intChan
if !ok {
fmt.Println("已从intChan读取所有数据")
break
}
for i := 2; i < num; i++ {
if num % i == 0 {
flag = false
break
}
if flag {
resultChan <- num
fmt.Printf("添加%v到resultChan\n", num)
}
}
}
fmt.Println("协程执行完毕")
exitChan <- true
}
func main() {
var wg sync.WaitGroup
intChan := make(chan int, 10000)
resultChan := make(chan int, 20000)
exitChan := make(chan bool, 8)
go writeData(intChan)
for i := 0; i < 8; i++ {
go judgData(intChan, resultChan, exitChan, wg.Done)
}
//go func() {
// for i := 0 ; i < 8 ; i ++ {
// key := <-exitChan
// fmt.Print(key)
// }
// close(resultChan)
//}()
wg.Add(8)
wg.Wait()
close(resultChan)
func() {
for i := 0; i < 8; i++ {
key := <-exitChan
fmt.Printf("key=%v\n", key)
}
}()
for {
res, ok := <-resultChan
if !ok {
break
}
fmt.Printf("素数=%d\n", res)
}
}
有疑问加站长微信联系(非本文作者)

就很奇怪,协程的确都已经结束了,但是计算好像没有结束:screen
wg传入进去是值传递副本,要么取地址 要么闭包方式共用外部wg
用管道阻塞实现了,这个打算先放一下
谢谢您
更新后的代码如下,可以正常工作,就是感觉不够优雅
5楼 @lihongbin 谢谢您,判断那里的确有问题,flag = true 写在循环体是因为每次循环结束后flag的值被置为false了,要改回来,下面一楼有我改好的代码,您可以帮忙看看有可以优化的么
给两个小建议吧
1、关闭的channel可以用for range 来循环,所以最后打印素数的时候,可以写成
2、写代码的时候,最好业务跟逻辑分离处理,比如把判断是否是素数的方法拆出来
其他的就是一些个人习惯了,练习项目都无所谓
学到了,谢谢您,还能请教您一个小问题么,我的一段代码
然而反序列的值无法写入到exitMes这个变量中