6

我正在尝试获取子进程的退出代码。在 Linux 和 FreeBSD 上,我可以这样:

[0] [ishpeck@kiyoshi /tmp]$ uname
FreeBSD
[0] [ishpeck@kiyoshi /tmp]$ cat tinker.c 
#include <stdio.h>
#include <sys/wait.h>

int main(void)
{
    FILE *proc = popen("ls", "r");
    printf("Exit code: %d\n", WEXITSTATUS(pclose(proc)));
    return 0;
}
[0] [ishpeck@kiyoshi /tmp]$ gcc tinker.c -o tinker
[0] [ishpeck@kiyoshi /tmp]$ ./tinker
Exit code: 0
[0] [ishpeck@kiyoshi /tmp]$ grep WEXITSTATUS /usr/include/sys/wait.h 
#define WEXITSTATUS(x)  (_W_INT(x) >> 8)

但是,在 OpenBSD 上,我收到了编译器的抱怨……

[0] [ishpeck@ishberk-00 /tmp]$ uname   
OpenBSD
[0] [ishpeck@ishberk-00 /tmp]$ cat tinker.c                                    
#include <stdio.h>
#include <sys/wait.h>

int main(void)
{
    FILE *proc = popen("ls", "r");
    printf("Exit code: %d\n", WEXITSTATUS(pclose(proc)));
    return 0;
}
[0] [ishpeck@ishberk-00 /tmp]$ gcc tinker.c -o tinker                          
tinker.c: In function 'main':
tinker.c:7: error: lvalue required as unary '&' operand
[1] [ishpeck@ishberk-00 /tmp]$ grep WEXITSTATUS /usr/include/sys/wait.h        
#define WEXITSTATUS(x)  (int)(((unsigned)_W_INT(x) >> 8) & 0xff)

我真的不在乎它是如何完成的,我只需要退出代码。

这让我相信我在 Mac 上也会遇到这个问题:http: //web.archiveorange.com/archive/v/8XiUWJBLMIKYSCRJnZK5#F4GgyRGRAgSCEG1

有没有更便携的方式来使用 WEXITSTATUS 宏?还是有更便携的选择?

4

3 回答 3

9

OpenBSD 的实现在其参数上WEXITSTATUS使用地址运算符(一元&),有效地要求其参数具有存储空间。您正在使用没有存储空间的函数的返回值调用它,因此编译器会抱怨。

目前尚不清楚 OpenBSD 是否WEXITSTATUS符合 POSIX,但可以通过将返回值pclose()赋给变量来轻松解决该问题:

    int status = pclose(proc);
    printf("Exit code: %d\n", WEXITSTATUS(status));
于 2012-12-02T22:50:01.730 回答
3

作为一些到达这里的人可能不会注意到的细节,BSD 目标代码需要该库:

#include <sys/wait.h>

我也编译到LinuxBSD,并且在编译到Linux(使用)WEXITSTATUS时不需要那个库(我不知道为什么)就可以正常工作,但是在编译到BSD时(使用)失败了。gccclang

于 2019-05-14T01:55:29.977 回答
0

如果您的应用程序死亡或以其他方式被终止,则返回状态是虚假的。您需要检查状态以查看退出值是否有效。请参阅 waitpid 的手册页。

if(WIFEXITED(status))
{
     use WEXITSTATUS(status);
} else if (WIFSIGNALED(status)) {
     use WTERMSIG(status);
} else {
     oh oh
}
于 2015-03-26T21:09:31.693 回答