system函数讨论

三个程序: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在其实现过程中调用了forkexecwaitpid,因此有三种返回值:

1)  如果fork失败或者waitpid返回除EINTR之外的出错,则system返回-1,而且errno中设置了错误类型。

2)  如果exec失败(表示不能执行exec),则其返回值如同shell执行了exit127)一样

3)  否则所有三个函数(forkexecwaitpid)都成功,并且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了。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值