gpt4 book ai didi

c 管道读取字符串为空

转载 作者:太空宇宙 更新时间:2023-11-04 04:33:14 25 4
gpt4 key购买 nike

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

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com