假设我有producer.go
和consumer.go
。 consumer.go
从UNIX命名管道读取,生产者写入命名管道。
正如预期的那样,如果您启动只有一个生产者或消费者程序,它会挂起,因为管道的另一侧没有读取器或写入器。
现在,如果我启动这两个程序,然后立即从消费者那里CTRL-C,生产者继续向管道发送数据,据我所知,这些数据的大小没有限制(我发送了80MB )
如果我再次启动消费者程序(当生产者仍然在运行时),它会开始从命名管道中提取数据,而不是在消费者程序未运行时我“错过”的数据。
我的问题是:当命名管道的阅读器断开连接时,发送到命名管道的数据会发生什么变化?
以下是我的consumer.go
和producer.go
计划:
consumer.go
package main
import (
"io"
"io/ioutil"
"log"
"os"
"syscall"
)
func main() {
syscall.Mkfifo("fifo0", 0777)
fp, err := os.OpenFile("fifo0", os.O_RDONLY, 0777)
if err != nil {
log.Fatalf("Could not open fifo0: %s", err)
}
tee := io.TeeReader(fp, os.Stdout)
ioutil.ReadAll(tee)
}
producer.go
package main
import (
"fmt"
"io"
"log"
"os"
"strings"
"time"
)
func main() {
dots := strings.Repeat(".", 990)
fifo, err := os.OpenFile("fifo0", os.O_WRONLY, 0777)
if err != nil {
log.Fatalf("Could not open fifo0: %s", err)
}
defer fifo.Close()
w := io.MultiWriter(os.Stdout, fifo)
for i := 0; i < 8000; i++ {
fmt.Fprintf(w, "%010d%s\n", i, dots)
time.Sleep(time.Millisecond * 10)
}
}
答案 0 :(得分:0)
FIFO至少需要一个源和一个目的地,数据才能在任何地方传输。一个读者独自等待从某人那里拉走,一个作家独自等待发送给某人。这样一来,一对一的管道就不会有间隙。
因此,如果您仍在尝试从断开连接或不存在的管道的一端进行读取或写入,则答案是数据无处可寻;管道被“阻塞”,不能容纳自己的数据。因此,这取决于您的代码如何处理这种情况。
在producer.go
中,即使不再存在连接,循环也继续运行。因为无论出于何种原因,Fprintf
或MultiWriter
均不会引发代码停止错误。在这种情况下,您可以在循环中添加检查,也可以为fifo
对象的断开连接添加事件处理程序。
似乎存在数据消失的空白的原因是因为循环继续迭代i
并生成无法发送的字符串。