【问题标题】:c pipe read string is emptyc管道读取字符串为空
【发布时间】:2015-11-17 14:31:52
【问题描述】:

我只想创建 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

我无法解决问题。

【问题讨论】:

  • "second" 太长,无法存储在只有 6 个元素的 myname 中。您应该指定元素的数量,例如 char myname[128] = "ALOAA";
  • 我认为read 会失败,因为fds[0],这是read 试图读取的内容,在读取之前已关闭。

标签: c string pipe fork


【解决方案1】:

但是我从 read() 函数中得到了空字符串 [...] 我无法解决问题。

@MikeCAT 通过他在 cmets 中观察到每个孩子在尝试读取之前关闭 fds[0] 解决了这个问题。没有其他文件被分配相同的FD,因此读取失败。你不测试失败。

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

  1. read() 将通过返回 -1 来指示失败,并且您的程序将通过尝试越界写入(至 readbuffer[-1])来响应。

  2. 1234563该数组)。

总体而言,您需要学习检查库函数调用的返回值是否存在错误情况的规则,至少在任何与是否发生错误有关的地方(对于大多数调用而言)都是如此。例如,您对pipe()fork()write() 的使用也存在此问题。在某些情况下你想检查printf()-family 函数的返回值,你通常想检查输入函数的返回值——不仅仅是read(),还有scanf()fgets()等。

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

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

【讨论】:

    猜你喜欢
    • 2014-02-12
    • 1970-01-01
    • 2020-03-24
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-12-14
    相关资源
    最近更新 更多