【转】Linux下(C/C++)使用system()函数一定要谨慎
1 #include <stdlib.h> 2 int system(const char *command);
system() executes a command specified in command by calling /bin/sh -c command, and returns after the command has been completed. During execution of the command, SIGCHLD will be blocked, and SIGINT and SIGQUIT will be ignored.
1 int system(const char * cmdstring) 2 { 3 pid_t pid; 4 int status; 5 6 if(cmdstring == NULL) 7 { 8 return (1); //如果cmdstring为空,返回非零值,一般为1 9 } 10 11 if((pid = fork())<0) 12 { 13 status = -1; //fork失败,返回-1 14 } 15 else if(pid == 0) 16 { 17 execl("/bin/sh", "sh", "-c", cmdstring, (char *)0); 18 _exit(127); // exec执行失败返回127,注意exec只在失败时才返回现在的进程,成功的话现在的进程就不存在啦~~ 19 } 20 else //父进程 21 { 22 while(waitpid(pid, &status, 0) < 0) 23 { 24 if(errno != EINTR) 25 { 26 status = -1; //如果waitpid被信号中断,则返回-1 27 break; 28 } 29 } 30 } 31 32 return status; //如果waitpid成功,则返回子进程的返回状态 33 }
仔细看完这个system()函数的简单实现,那么该函数的返回值就清晰了吧,那么什么时候system()函数返回0呢?只在command命令返回0时。
1 int status; 2 if(NULL == cmdstring) //如果cmdstring为空趁早闪退吧,尽管system()函数也能处理空指针 3 { 4 return XXX; 5 } 6 status = system(cmdstring); 7 if(status < 0) 8 { 9 printf("cmd: %s\t error: %s", cmdstring, strerror(errno)); // 这里务必要把errno信息输出或记入Log 10 return XXX; 11 } 12 13 if(WIFEXITED(status)) 14 { 15 printf("normal termination, exit status = %d\n", WEXITSTATUS(status)); //取得cmdstring执行结果 16 } 17 else if(WIFSIGNALED(status)) 18 { 19 printf("abnormal termination,signal number =%d\n", WTERMSIG(status)); //如果cmdstring被信号中断,取得信号值 20 } 21 else if(WIFSTOPPED(status)) 22 { 23 printf("process stopped, signal number =%d\n", WSTOPSIG(status)); //如果cmdstring被信号暂停执行,取得信号值 24 }
到于取得子进程返回值的相关介绍可以参考另一篇文章:http://my.oschina.net/renhc/blog/35116
system()函数用起来很容易出错,返回值太多,而且返回值很容易跟command的返回值混淆。这里推荐使用popen()函数替代,关于popen()函数的简单使用也可以通过上面的链接查看。
popen()函数较于system()函数的优势在于使用简单,popen()函数只返回两个值:
成功返回子进程的status,使用WIFEXITED相关宏就可以取得command的返回结果;
失败返回-1,我们可以使用perro()函数或strerror()函数得到有用的错误信息。
这篇文章只涉及了system()函数的简单使用,还没有谈及SIGCHLD、SIGINT和SIGQUIT对system()函数的影响,事实上,之所以今天写这篇文章,是因为项目中因有人使用了system()函数而造成了很严重的事故。现像是system()函数执行时会产生一个错误:“No child processes”。
关于这个错误的分析,感兴趣的朋友可以看一下:http://my.oschina.net/renhc/blog/54582