C到Python管道-如何检测读者访问
我正在使用 C 程序写入命名管道并使用 Python 程序读取它。
如果我停止 Python 程序(阅读器),那么编写器会自行停止,尽管这是在 while(1) 循环中。为什么会发生这种情况?是无声的崩溃吗?
第二个问题,如果我想检测读取器何时断开连接,我应该怎么做。我的理想情况是检测断开连接然后继续空闲(即停止发送任何内容)并在读者回来后恢复。
下面的玩具代码。
作家 (C):
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
int fd;
// FIFO file path
char * myfifo = "/tmp/myfifo";
// Creating the named file(FIFO)
// mkfifo(<pathname>, <permission>)
mkfifo(myfifo, 0666);
char arr1[80];
while (1)
{
// Open FIFO for write only
fd = open(myfifo, O_WRONLY);
// Take an input from user.
fgets(arr1, 80, stdin);
// Write the input on FIFO
// and close it
write(fd, arr1, strlen(arr1)+1);
close(fd);
}
return 0;
}
阅读器(Python)
f = open("/tmp/myfifo")
while 1:
print(f.readline(), end = "")
f.close()
回答
当您停止读取程序时,写入器将SIGPIPE在尝试写入管道时收到信号。此信号的默认处置是终止进程。
如果要检测此情况,请使用signal()或sigaction()将处置更改为SIG_IGN。然后写入管道将报告EPIPE错误。
此外,您不应该每次通过循环关闭并重新打开管道。开头打开一次,结尾关闭。关闭管道会导致读取器获得 EOF,之后它将无法读取任何内容。
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <signal.h>
int main()
{
int fd;
// FIFO file path
char * myfifo = "/tmp/myfifo";
// Creating the named file(FIFO)
// mkfifo(<pathname>, <permission>)
mkfifo(myfifo, 0666);
// Open FIFO for write only
fd = open(myfifo, O_WRONLY);
signal(SIGPIPE, SIG_IGN);
char arr1[80];
while (1)
{
// Take an input from user.
fgets(arr1, 80, stdin);
// Write the input on FIFO
// and close it
int res = write(fd, arr1, strlen(arr1)+1);
if (res < 0) {
perror("write");
break;
}
}
close(fd);
return 0;
}
当您停止编写程序时,读取器在尝试从管道中读取时将获得 EOF。当f.readline()到达 EOF 时,它返回一个空字符串。您的 Python 脚本不会对此进行检查,因此它会无限循环。
将读取器更改为:
with open("/tmp/myfifo") as f:
while True:
line = f.readline()
if not line:
break
print(line, end = "")
然后当管道关闭时循环将终止。