一、popen函数
1.1 函数原型
#include <stdio.h>
FILE *popen(const char *command, const char *type);
int pclose(FILE *stream);
1.2 功能描述
- popen()函数通过创建一个管道,调用==fork()==产生一个子进程,执行一个shell以运行命令来开启一个进程。可以通过管道执行标准输入输出操作
- 这个管道必须由pclose()函数关闭,而不是fclose()函数(若使用fclose则会产生僵尸进程)。pclose()函数关闭标准I/O流,等待命令执行结束,然后返回shell的终止状态。如果shell不能被执行,则pclose()返回的终止状态与shell已执行exit一样
1.3 参数与返回值描述
1.3.1 参数
- command参数是一个指向以NULL结束的shell命令字符串的指针。这行命令将被传到bin/sh并使用-c标志,shell将执行这个命令。
- type参数只能是读或写中的一种,得到的返回值(标准I/O流)也具有和type相应的只读或只写类型。如果type是"r"则文件指针连接到command的标准输出;如果type是"w"则文件指针连接到command的标准输入。
1.3.2 返回值
- popen()的返回值是个标准I/O流,必须由pclose来终止。前面提到这个流是单向的(只能用于读或写)。向这个流写内容相当于写入该命令的标准输入,命令的标准输出和调用popen()的进程相同;与之相反的,从流中读数据相当于读取命令的标准输出,命令的标准输入和调用popen()的进程相同。
- 如果调用fork()或pipe()失败,或者不能分配内存将返回NULL,否则返回标准I/O流。popen()没有为内存分配失败设置errno值。如果调用fork()或pipe()时出现错误,errno被设为相应的错误类型。如果type参数不合法,errno将返回EINVAL。
二、system函数
2.1 函数原型
#include <stdlib.h>
int system(const char * string);
2.2 功能描述
- system()会调用fork() 产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令。
- 此命令执行完后随即返回原调用的进程。
- 在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。
2.3 返回值描述
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.
三、区别
- popen是不堵塞的,也就是说不会等待子进程的结束并杀死子进程,即不会管理进程。这样就需要我们人为的去杀死或忽略子进程等操作。还有就是popen会将执行的结果返回到buf中。
- system是堵塞的,会自动对进程进行管理,无需我们再去对进程进行管理。另外,system不会返回执行的结果,只是会返回执行是否成功。
参考:https://blog.csdn.net/yaowangII/article/details/82997102