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