三个程序:test1.c test2.c test3.c
gcc test1.c –o test1
在test2.c中调用system(“./test1”);
gcc test2.c –o test2
在test2.c中调用popen(“./test2”,”r”);
gcc test3.c –o test3
./test3发现test2.cz中的system返回-1,但是test1执行了。查看system的源码
可以发现,system在其实现过程中调用了fork、exec和waitpid,因此有三种返回值:
1) 如果fork失败或者waitpid返回除EINTR之外的出错,则system返回-1,而且errno中设置了错误类型。
2) 如果exec失败(表示不能执行exec),则其返回值如同shell执行了exit(127)一样
3) 否则所有三个函数(fork、exec和waitpid)都成功,并且system的返回值是shell的终止状态,其格式已在waitpid中说明。
那么我这里system返回-1,应该是waitpid的时候返回的,于是将waitpid改成wait,发现system的返回值是正确的。
查看linux源代码 unistd.h 我们会发现,其实 wait 就是经过包装的 waitpid:
static inline pid_t wait(int*wait_stat)
{
return waitpid(-1,wait_stat,0);
}
waitpid的返回值比wait稍微复杂一些,一共有3种情况:
1. 当正常返回的时候,waitpid返回收集到的子进程的进程ID;
2. 如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;
3. 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;
当pid所指示的子进程不存在,或此进程存在,但不是调用进程的子进程,waitpid就会出错返回,这时errno被设置为ECHILD。
这里waitpid返回出错应该是等到了pid的父进程退出,试着让父进程sleep(10),就ok了。