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 = "")

然后当管道关闭时循环将终止。


以上是C到Python管道-如何检测读者访问的全部内容。
THE END
分享
二维码
< <上一篇
下一篇>>