1. 首页
  2. 主题
  3. Go问与答

求助,关于waitgroup产生死锁问题

zhuting233 · · 2010 次点击
``` package main import ( "bufio" "fmt" "math/rand" "os" "strconv" "sync" ) var path string = "./num.txt" func Write() { file, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0755) defer file.Close() if err != nil { fmt.Println(err) } writer := bufio.NewWriter(file) for i := 0; i < 1000; i++ { x := rand.Intn(2000) writer.WriteString(strconv.Itoa(x) + "\n") } writer.Flush() } var wg sync.WaitGroup func Read() chan int { file, err := os.Open(path) if err != nil { fmt.Println(err) } defer file.Close() scanner := bufio.NewScanner(file) ch := make(chan int, 1000) for i := 0; i < 3; i++ { wg.Add(1) go func() { defer wg.Done() for scanner.Scan() { t, _ := strconv.Atoi(scanner.Text()) // fmt.Println(t) ch<-t } }() } wg.Wait() close(ch) return ch } func main() { Write() ch := Read() fmt.Println(len(ch)) } ``` 有时候会产生死锁,大部分情况下通道中数据长度也不对。 本人小白,思考很久,不知道为啥会出现问题,求助~ 谢谢各位
buguang01
https://github.com/buguang01
我理解,你是写了一个1000行的文件,然后想并发读这个文件写到ch中;但是你发现写入ch的数据长度不对,还有时候会卡在写入ch的地方,对吧? 原因是这样的,scanner这个并不是协程安全的,你想一下,三个协程同时 scanner.Scan()然后scanner.Text()的时候,他们是分别获得三行,还是获得同一行呢?因为scanner.Scan()的时候没有锁,导致scanner里面的过程变量的值变得不可预测,这会导致读出来的数据不是正确的。所以你的这个程序会出现你现在发现的问题。
#2
更多评论
这个不是waitgroup死锁的问题,是你读文件的方式有问题 感觉你的这个写法很奇怪,没看懂你的目的是什么。
#1
buguang01
https://github.com/buguang01
你在编译的时候加上-race 就很明显看出来了。 go build -race main.go 然后运行main.exe 你就能看到警告了。
#3

用户登录

没有账号?注册

今日阅读排行

    加载中

一周阅读排行

    加载中