```
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))
}
```
有时候会产生死锁,大部分情况下通道中数据长度也不对。
本人小白,思考很久,不知道为啥会出现问题,求助~ 谢谢各位
你是不是想着,我有一个文件,想多开几个协程一起读,是不是可以更快的读入文件?
这个问题其实可以多几个方面考虑;
---
1. 同一个文件,可以使用只读的方式,打开多个句柄,然后每个句柄从不同的位置开始读;
2. 如果读出后的处理操作很费时间的话,可以同一个文件使用一个句柄,读的时候需要加上并发锁,并发处理;
#4
更多评论
我理解,你是写了一个1000行的文件,然后想并发读这个文件写到ch中;但是你发现写入ch的数据长度不对,还有时候会卡在写入ch的地方,对吧?
原因是这样的,scanner这个并不是协程安全的,你想一下,三个协程同时 scanner.Scan()然后scanner.Text()的时候,他们是分别获得三行,还是获得同一行呢?因为scanner.Scan()的时候没有锁,导致scanner里面的过程变量的值变得不可预测,这会导致读出来的数据不是正确的。所以你的这个程序会出现你现在发现的问题。
#2