- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我刚刚学习 fork() 在 C 中的工作原理。这个想法是生成 3 个子进程,每个子进程向父进程发送一些信息。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main()
{
int fd[2];
int pid[3] = {0,0,0};
int status = 0;
for (int i = 0; i < 3; i++)
{
pipe(fd);
pid[i] = fork();
if (pid[i] < 0)
{
return -1;
}
if (pid[i] == 0)
{
close(fd[0]);
char *arr = malloc(sizeof(char));
sprintf(arr, "%i", i);
write(fd[1], arr, 1);
exit(0);
}
}
for(int j=0; j < 3; j++)
{
close(fd[1]);
if (pid[j] > 0)
{
sleep(0);
pid[j] = wait(&status);
char *out = malloc(20 *sizeof(char));
read(fd[0], out, 6);
printf("%s\n", out);
free(out);
printf("I am the parent\n");
}
}
}
预期的输出是:
1
I am the parent
2
I am the parent
3
I am the parent
实际输出是: 2 我是家长 2 我是家长 2 我是家长
为什么会发生这种情况?
最佳答案
几点:
您在第一个 pipe(fd)
循环中调用 for
三次,并且在创建三个管道时,您仅保存对其中一个管道的引用。因此,在第二个 for
循环中,您将从每次创建的第三个管道中读取数据。您应该有一个数组来存储对所有三个管道的引用。
您应该检查所有可能失败的系统调用的返回结果。由于上面的第 1 点,如果您执行此操作,close(fd[1])
将在三分之二的情况下失败,并且它会提醒您出现了问题。检查系统调用的返回不仅是为了防止出现不太可能发生的错误,而且还可以帮助您进行调试,因为它们失败的最可能原因是您做错了什么,就像这里的情况一样。
这里绝对不需要使用 malloc()
- 常规的 char
数组就可以了。此外,当您需要至少两个字符(即单个数字和终止空字符)时,malloc(sizeof(char));
会为一个字符分配空间。此外,您应该始终检查 malloc()
的返回,因为它可能会失败。另外,根据定义,sizeof(char)
始终是 1
,因此它总是多余的。
要获得所需的输出,您应该将 1
添加到 i
,否则 i
将是 0
,然后是 1
,然后是 2
,但您的示例输出表明您想要 1
,然后是 2
和 3
。
waitpid()
比 wait()
更好,因为您想等待特定进程。同样,当您使用它时,您的 sleep()
调用是多余的。
虽然在退出之前没有必要在此处对管道进行 close()
,但有时这样做会很有帮助,因为如果您做错了什么,它可能会引起您注意。
您的 if (pid[j] > 0)
检查是不必要的,因为如果 fork()
失败或者它是 0
,您已经终止,所以您已经知道当您到达这里时它将大于 0
。
如果您不打算使用 status
变量来检索进程的退出状态,则不需要它 - 您只需将 NULL
传递到 wait()
或 waitpid()
即可。
小问题,但您不需要为第二个 j
循环使用不同的变量名称(即 for
),因为第一个 i
循环中 for
的范围仅限于该循环。如果您打算使用 i
作为循环计数器的通用名称,那么您不妨在任何可以使用的地方使用它。
return EXIT_FAILURE
比 return -1
更好,并且当您检索退出状态时,该值将转换为 255
。
这在现实中不太可能成为问题,但 fork()
返回类型 pid_t
,而不是类型 int
,因此最好将 pid
设为该类型的数组。
这是一些修改后的代码:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(void)
{
int fd[3][2];
pid_t pid[3];
for ( int i = 0; i < 3; ++i ) {
if ( pipe(fd[i]) == -1 ) {
perror("pipe() failed");
return EXIT_FAILURE;
}
if ( (pid[i] = fork()) == -1 ) {
perror("fork() failed");
return EXIT_FAILURE;
}
else if ( pid[i] == 0 ) {
if ( close(fd[i][0]) == -1 ) {
perror("close() failed");
return EXIT_FAILURE;
}
char arr[100];
sprintf(arr, "%d", i + 1);
if ( write(fd[i][1], arr, 1) == -1 ) {
perror("write() failed");
return EXIT_FAILURE;
}
if ( close(fd[i][1]) == -1 ) {
perror("close() failed");
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
else {
if ( close(fd[i][1]) == -1 ) {
perror("close() failed");
return EXIT_FAILURE;
}
}
}
for ( int i = 0; i < 3; ++i ) {
if ( waitpid(pid[i], NULL, 0) == -1 ) {
perror("waitpid() failed");
return EXIT_FAILURE;
}
char out[100] = {0};
if ( read(fd[i][0], out, 99) == -1 ) {
perror("read() failed");
return EXIT_FAILURE;
}
printf("%s\nI am the parent\n", out);
if ( close(fd[i][0]) == -1 ) {
perror("close() failed");
return EXIT_FAILURE;
}
}
return EXIT_SUCCESS;
}
输出:
paul@horus:~/src/sandbox$ ./pipe
1
I am the parent
2
I am the parent
3
I am the parent
paul@horus:~/src/sandbox$
关于C 多处理/管道,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28289157/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!