system()函数功能强大,我对linux中的实现比较了解,具体分析这个,windows中的类似就不详解了。
//好了,先看linux版system函数的源码:
#include
#include
#include
#include
int system(const char * cmdstring)
{
pid_t pid;
int status;
if(cmdstring == NULL){
return (1);
}
if((pid = fork())<0){
status = -1;
}
else if(pid == 0){
execl("/bin/sh", "sh", "-c", cmdstring, (char *)0);
-exit(127); //子进程正常执行则不会执行此语句
}
else{
while(waitpid(pid, &status, 0) < 0){
if(errno != EINTER){
status = -1;
break;
}
}
}
return status;
}
分析一下原理: 当system接受的命令为NULL时直接返回,否则fork出一个子进程,因为fork在两个进程:父进程和子进程中都返回,这里要检查返回的 pid,fork在子进程中返回0,在父进程中返回子进程的pid,父进程使用waitpid等待子进程结束,子进程则是调用execl来启动一个程序代替自己,execl("/bin/sh", "sh", "-c", cmdstring,(char*)0)是调用shell,这个shell的路径是/bin/sh,后面的字符串都是参数,然后子进程就变成了一个 shell进程,这个shell的参数是cmdstring,就是system接受的参数。在windows中的shell是command,想必大家很熟悉shell接受命令之后做的事了。 windows中的情况也类似,就是execl换了个又臭又长的名字,参数名也换的看了让人发晕的。
system(执行shell 命令)
相关函数 fork,execve,waitpid,popen
表头文件 #include<stdlib.h>
定义函数 int system(const char * string);
函数说明 system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。 返回值 成功,则返回进程的状态值;当sh不能执行时,返回127;失败返回-1。
popen
函数原型:
#include “stdio.h”
FILE popen( const char* command, const char* mode );
参数说明:
command: 是一个指向以 NULL 结束的 shell 命令字符串的指针。这行命令将被传到 bin/sh 并使用 -c 标志,shell 将执行这个命令。
mode: 只能是读或者写中的一种,得到的返回值(标准 I/O 流)也具有和 type 相应的只读或只写类型。如果 type 是 “r” 则文件指针连接到 command 的标准输出;如果 type 是 “w” 则文件指针连接到 command 的标准输入。
返回值:
如果调用成功,则返回一个读或者打开文件的指针,如果失败,返回NULL,具体错误要根据errno判断
作用:
popen() 函数用于创建一个管道:其内部实现为调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程这个进程必须由 pclose() 函数关闭。
int pclose (FILE* stream);
参数说明:
stream:popen返回的文件指针
返回值:
如果调用失败,返回 -1
比system/exec族函数在应用中的好处: 可以获取运行的输出结果
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main()
{
FILE *fd;
char ret[1024]={0};
//使用popen函数调用ps指令,将内容存到ret中并且打印,popen函数能将运行结果返回
fd = popen("ps","r");
int nread = fread(ret,1,1024,fd);
printf("read ret %d byte,ret = %s\n",nread,ret);
return 0;
}
结果:
CLC@Embed_Learn:~/Linux/course$ ./a.out
read ret 141 byte,ret =
PID TTY TIME CMD
24209 pts/0 00:00:02 bash
25506 pts/0 00:00:00 a.out
25507 pts/0 00:00:00 sh
25508 pts/0 00:00:00 ps