c 管道读取字符串为空

标签 c string pipe fork

我只想创建 2 个新的分支(子进程),它们将依次 放置它们的名称。所以首先他们需要一些管道中的字符串来检查一些东西。让我们看看代码:

char myname[] = "ALOAA";

int main ()
{

    int fds[2];
    pid_t pid;
    pipe(fds);
    pid = fork();

    if(pid == 0)
    {
        strcpy(myname, "first");
    }
    else
    {
        pid = fork();
        if(pid == 0)
        {
            strcpy(myname, "second");
        }
    }




    if(strcmp(myname, "ALOAA") != 0)
    {

        char    readbuffer[1025];

        int i;
        for (i = 0; i < 2 ; i++)
        {
            //printf("%s\n", myname);
            close(fds[0]);
            write(fds[1], myname, strlen(myname));
            while(1)
            {
                close(fds[1]);
                int n = read(fds[0], readbuffer, 1024);
                readbuffer[n] = 0;
                printf("%s-alihan\n", readbuffer);
                if(readbuffer != myname)
                    break;
                sleep(1);
            }
            //printf("%s\n", myname);
        }



    }




    return 0;

}

所以第一个进程会将她的名字写入管道。之后,将检查管道中是否有任何新字符串。第二个也一样。但是我从 read() 函数中得到了空字符串。所以它打印成那样

-alihan
-alihan

我无法解决问题。

最佳答案

However I got empty string from read() function [...] I couldn't get the problem.

@MikeCAT 通过他在评论中的观察指出了这个问题,即每个 child 在尝试读取它之前都会关闭 fds[0]。之间没有其他文件分配相同的 FD,因此读取失败。您不测试失败。

不测试读取失败是一个严重的问题,因为您的程序不仅无法识别它——结果会表现出未定义的行为。出现这种情况(至少)有两个原因:

  1. read() 将通过返回 -1 指示失败,您的程序将通过尝试越界写入(到 readbuffer[-1]).

  2. 如果我们忽略由 (1) 产生的 UB,此后我们仍然有程序从完全未初始化的数组 readbuffer 读取(因为 read()调用或赋值都不会设置该数组的任何元素的值)。

总的来说,您需要学习检查库函数调用的返回值以了解错误情况的原则,至少在任何与错误是否发生有关的地方(对于大多数调用而言)。例如,您对 pipe()fork()write() 的使用也出现了这个问题。在某些情况下,您想要检查 printf() 系列函数的返回值,并且您通常想要检查输入函数的返回值——而不仅仅是 read(),但是 scanf()fgets() 等。

第三,您对 read()write() 的使用不正确。您犯了一个常见的错误,即假设(成功时)write() 将可靠地写入所有指定的字节,而 read() 将读取所有已写入的字节, 直到指定的缓冲区大小。虽然这在实践中通常适用于通过管道交换短消息,但不能保证。一般来说,write() 可能只执行部分写入,而 read() 可能只执行部分读取,原因不明且不可预测。

要成功写入,通常必须准备好在循环中重复调用 write(),使用返回值来确定从何处(或是否)开始下一次写入。要成功读取完整的消息,通常必须准备类似于在循环中重复 read() 调用,直到已将所需数量的字节读入缓冲区,或者直到满足其他一些终止条件,例如作为到达文件的末尾。我想您不会忘记,许多形式的这种情况需要预先了解要读取的字节数。

关于c 管道读取字符串为空,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33759495/

相关文章:

我们可以将 bzip2 block 大小减少到 100KB 以下吗?

c - 如何用C语言在客户端-服务器程序中建立双向链接?

c - 如何定义和 typedef 一个包含指向自身的指针的结构?

c - 无法创建会反转字符串的程序

bash - 将左侧故障通知管道的右侧?

c - 我如何将 "tell"告诉 C 编译器,代码不应该被优化掉?

java - Java中如何分割长字符串?

linux - 如何在变量中存储 echo |削减结果?

c - 我应该在 poll() 循环中调用 close()

node.js - 未知方法 process.openStdin()