曾经的曾经,被system()函数折磨过,之所以这样,是因为对system()函数了解不够深入。只是简单的知道用这个函数执行一个系统命令,这远远不够,它的返回值、它所执行命令的返回值以及命令执行失败原因如何定位,这才是重点。当初因为这个函数风险较多,故抛弃不用,改用其他的方法。这里先不说我用了什么方法,这里必须要搞懂system()函数,因为还是有很多人用了system()函数,有时你不得不面对它。

 
先来看一下system()函数的简单介绍:
 
1 #include <stdlib.h>
2 int system(constchar *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.

  system()函数调用/bin/sh来执行参数指定的命令,/bin/sh 一般是一个软连接,指向某个具体的shell,比如bash,-c选项是告诉shell从字符串command中读取命令;
  在该command执行期间,SIGCHLD是被阻塞的,好比在说:hi,内核,这会不要给我送SIGCHLD信号,等我忙完再说;
  在该command执行期间,SIGINT和SIGQUIT是被忽略的,意思是进程收到这两个信号后没有任何动作。
 
再来看一下system()函数返回值:
The value returned is -1 on error (e.g. fork(2) failed), and the return status of the command otherwise. This latter return status is in the format specified in wait(2). Thus, the exit code of the command will be WEXITSTATUS(status). In case /bin/sh could not be executed, the exit status will be that of a command that does exit(127).
If the value of command is NULL, system() returns nonzero if the shell is available, and zero if not.
为了更好的理解system()函数返回值,需要了解其执行过程,实际上system()函数执行了三步操作: 
1.fork一个子进程;
2.在子进程中调用exec函数去执行command;
3.在父进程中调用wait去等待子进程结束。
  对于fork失败,system()函数返回-1。
  如果exec执行成功,也即command顺利执行完毕,则返回command通过exit或return返回的值。
  (注意,command顺利执行不代表执行成功,比如command:"rm debuglog.txt",不管文件存不存在该command都顺利执行了)
  如果exec执行失败,也即command没有顺利执行,比如被信号中断,或者command命令根本不存在,system()函数返回127.
如果command为NULL,则system()函数返回非0值,一般为1.
 
看一下system()函数的源码
  看完这些,我想肯定有人对system()函数返回值还是不清楚,看源码最清楚,下面给出一个system()函数的实现:
 
01 int system(constchar * cmdstring)
02 {
03     pid_t pid;
04     intstatus;
05  
06 if(cmdstring == NULL)
07 {
08     return(1); //如果cmdstring为空,返回非零值,一般为1
09 }
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     returnstatus; //如果waitpid成功,则返回子进程的返回状态
33 }

 

仔细看完这个system()函数的简单实现,那么该函数的返回值就清晰了吧,那么什么时候system()函数返回0呢?只在command命令返回0时。
 
看一下该怎么监控system()函数执行状态
  这里给我出的做法:
01 int status;
02 if(NULL == cmdstring)//如果cmdstring为空趁早闪退吧,尽管system()函数也能处理空指针
03 {
04     returnXXX;
05 }
06 status = system(cmdstring);
07 if(status < 0)
08 {
09     printf("cmd: %s\t error: %s", cmdstring,strerror(errno));// 这里务必要把errno信息输出或记入Log
10     returnXXX;
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 }

 

到于取得子进程返回值的相关介绍可以参考另一篇文章:

 

system()函数用起来很容易出错,返回值太多,而且返回值很容易跟command的返回值混淆。这里推荐使用popen()函数替代,关于popen()函数的简单使用也可以通过上面的链接查看。

popen()函数较于system()函数的优势在于使用简单,popen()函数只返回两个值:

成功返回子进程的status,使用WIFEXITED相关宏就可以取得command的返回结果;
失败返回-1,我们可以使用perro()函数或strerror()函数得到有用的错误信息。

这篇文章只涉及了system()函数的简单使用,还没有谈及SIGCHLD、SIGINT和SIGQUIT对system()函数的影响,事实上,之所以今天写这篇文章,是因为项目中因有人使用了system()函数而造成了很严重的事故。现像是system()函数执行时会产生一个错误:“No child processes”。

关于这个错误的分析,将会专门写一篇文章分析。

 

 

2012-04-14 任洪彩 qdurenhongcai@163.com

转载请注明出处。

 

#include<iostream> //标准输入输出流,不同于stdio.h和cstdio

#include<unistd.h> //unix标准头文件,win环境下没有
#include<cstring> //c字符
#include<string> //c++字符,不同于cstring和string.h两个头文件
#include<cstdlib> //c语言的stdlib.h,这个里面使用了其中的system
typedef char* charPtr; //定义一个字符指针,以定义动态数组
using namespace std; //标准命名空间
int main( int argc, char* argv[] )
{
  string a; //定义c++字符串a
  a = " /home/fruit/bin/dx 哈哈,我终于成功啦!!!"; //给a赋值,也就是我想要执行的命令。这个命令在我系统的实际意思是执行的写的飞信伪脚本(用perl写的),然后发送“哈哈,我终于成功啦!!!”这句话给我自己。
  int n; //定义变量n,目的是取a字符串长度
  n = a.length(); //n取值
  charPtr b; //定义字符指针b
  b = new char[n+1]; //为b分配内存空间,为a的长度加一,因为c字符要以\0结尾
  a.data(); //将a以c字符返回
  a.c_str(); //将a加上\0
  a.copy(b , n, 0); //将a复制给b,其中b是指针类型,n是复制长度,0是index位置
  cout << " a " << a << endl //以下三行很简单,就是输出一些东西,让我自己了解程序运行情况,可以删除的
  << " b " << b << endl
  << " n " << n << endl;
  system ( b ) ; //前面所有为它服务,用system调用
}
我的初衷是想要将一个string的字符串作为一个新的进程执行。由于system只能支持char字符,所以需要将c++的string字符串转换为c的char字符组。
然后实际上我们可以不使用system()函数,可以直接调用exec函数族。
exec函数族中是有execve是系统调用,其余的均是以execve为基础的函数调用。
对于exec函数族是包含在unistd.h头文件中的。具体的exec函数族我就不在这里详述了,下次有机会有时间在讲一讲。
 

 

system是用shell来调用程序

而exec是直接让你的程序代替用来的程序运行
看一下,下面的例子,因为这里是perl组所以就用perl来具例子,实际情况下,在C中也差不多..
---------------------------------
#!/usr/bin/perl
#example1.pl
system("you_program");
print "You can see me! ";
---------------------------------
---------------------------------
#!/usr/bin/perl
#example2.pl
exec("you_program");
print "You can't see me! ";
---------------------------------
在example1.pl中,在你的程序执行完毕以后,会执行print语句。
在example2.pl中,由于exec将程序your_program代替了本身,
因此程序不再会执行print语句。
在Linux下,exec通常会和fork语句一起用。
看下面的这个例子
--------------------------------------------
#!/usr/bin/perl
#example for fork() and exec()
$pid = fork();
if ($pid < 0) {
#fork函数出错
print "ERROR ";
exit(-1);
} else if ($pid == 0) {
#这里是子进程
print "I'm son! ";
#执行其它的程序
exec("your_program");
} else {
#这里是父进程
print "i'm father! ";
#等待子进程结束后返回
wait();
exit(0);
}

转自