php中popen,exec,system,passthru到底有多大区别

原创 2011年01月14日 13:22:00

今天  疼,第一次翻了php的源代码,因为想搞清楚如题所示的东西,虽然没有把源码看得太明白,但是我要的答案已经有了。

好了,先看看部分源码:

no

这个可以看出exec,system,passthru都是调用的php_exec_ex函数来实现的,只是有个参数不同,所以对他们的处理会有些微妙的差别。再看看php_exec_ex

no

我知道php_exec_ex又调用了php_exec:

no

跳一段(跳过几行参数分析的代码):

no

现在能看出这玩意是用VCWD_POPEN函数去执行的命令,而VCWD_POPEN是调用系统的popen函数,据高人说:

系统popen函数的执行流程又是怎么样的呢?

1、使用pipe系统调用创建一个管道
2、创建一个新的进程,将拷贝标准输入或者标准输出到文件 描述符。
3、关闭pipe返回的文件描述符。
4、调用execve系统调用来执行命令。

最后php_exec执行了:

no

 

把管道关闭了。

 

好了,现在在看看php的popen,之所以分开说是因为php把他归纳到file.c里边,没和上边三个在一块儿,当然它本身也有点不一样。

no

(此处省略n行,具体自己翻代码)

no

(此处又省n行)

no

好,到此popen函数结束了。

可以看出这四个函数实际上都是调用的VCWD_POPEN去执行命令,exec/system/passthru之间的区别我不感兴趣,我感兴趣的是popen和他们三的区别:

1.popen的mode参数是可以手动设置的"r"/"w",另外三只能是"r"(windows下是"rb")

2.popen没有读取和关闭管道,另外三是立即读取管道然后关闭管道

所以popen不会等待程序执行完成,而另外三必须等程序执行完成(推向后台的程序除外)。那如果popen打开了一个推向后台(mycmd &)的程序,然后又没有重定向stdout/stderr,那么当mycmd产生输出的时候,也就是想管道写入的时候,php引擎会收到SIGPIPE信号,然后中断程序,简单的说就是:程序在写一个读端被关闭的管道,会产生SIGPIPE信号。如果没推向后端,php会等待他调用的系统popen打开的命令返回才能收工,所以会卡在pclose或者没有pclose就在你的程序最后。

 

 

添加一段C里边popen说明:

FILE *popen(const char *command, const char *type); 

int pclose(FILE *stream); 

描述 

popen() 函数 用 创建管道 的 方式 启动 一个 进程, 并调用 shell. 因为 管道 是被定义成 单向的, 所以 type 参数 只能 定义成 只读 或者 只写, 不能是 两者同时, 结果流 也相应的 是 只读 或者 只写. 
command 参数 是 一个 字符串指针, 指向的 是 一个 以 null 结束符 结尾的字符串, 这个 字符串 包含 一个 shell 命令. 这个命令 被送到 /bin/sh 以 -c 参数 执行, 即由 shell 来执行. type 参数 也是 一个 指向 以 null 结束符 结尾的 字符串的指针, 这个字符串 必须是 'r' 或者 'w’ 来指明 是 读还是写.

popen()  函数 的 返回值 是 一个 普通的 标准I/O流, 它只能用 pclose()  函数 来关闭, 而不是 fclose() . 函数. 向 这个流 的 写入 被转化为 对 command 命令的 标准输入; 而 command 命令的 标准输出 则是和 调用popen() , 函数 的 进程 相同,除非 这个 被command命令 自己 改变. 相反的, 读取 一个 “被popen了的” 流, 就相当于 读取 command 命令的 标准输出, 而 command 的 标准输入 则是和 调用 popen , 函数的 进程 相同.

注意, popen  函数的 输出流 默认是 被全缓冲的.

 

pclose  函数 等待 相关的进程 结束 并返回 一个 command 命令的 退出状态, 就像 wait4  函数 一样 

 

PHP如何执行Linux系统命令呢?system();exec();popen();函数

PHP执行linux系统命令的三个基本函数 system();函数 说明:执行外部程序并显示输出资料。 语法:string system(string command, int [ret...
  • zhuoxiong
  • zhuoxiong
  • 2012年03月13日 11:58
  • 3504

PHP exec 函数 及 system,exec,passthru区别用法详解

exec     执行外部程式。           语法       :   string   exec(string   command,   string   [array],   int ...
  • yangjun07167
  • yangjun07167
  • 2010年05月18日 11:26
  • 23110

popen——php多进程利器

标题有点儿夸张,   我(们,本来想用们的,还会去掉了)运行系统命令常常用exec,system之类的, 但是今天发现了proc_open和popen,proc_open自称比popen多一些功...
  • leinchu
  • leinchu
  • 2012年11月16日 15:26
  • 7246

用popen实现在PHP代码异步调用服务器端的shell脚本

先看代码: Php代码   //php code ...      //然后执行一个服务器端操作   pclose(popen("/usr/local/php...
  • u010412301
  • u010412301
  • 2017年02月28日 20:23
  • 1451

popen实现PHP并发方法

用途: 当业务需要调用一个比较耗时外部资源时使用, 最常见的是从外部API获取相关内容使用方法: 1. 建立外部资源脚本。 如示例中使用t.sh。 当然, 也可以直接使用curl等命令行2. 准备好参...
  • binger819623
  • binger819623
  • 2009年08月15日 11:46
  • 1095

【转】php中popen,exec,system,passthru到底有多大区别

今天  疼,第一次翻了php的源代码,因为想搞清楚如题所示的东西,虽然没有把源码看得太明白,但是我要的答案已经有了。 好了,先看看部分源码: 这个可以看出exec,system,p...
  • weiyong1999
  • weiyong1999
  • 2012年09月20日 22:38
  • 5262

C程序调用shell脚本共有三种方式:system()、popen()、exec系列函数

1)system(shell命令或shell脚本路径);          执行过程:system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数stri...
  • luokehua789789
  • luokehua789789
  • 2016年11月10日 17:27
  • 6804

Linux的system()和popen()差异

Linux的system()和popen()差异1. system()和popen()简介在linux中我们可以通过system()来执行一个shell命令,popen()也是执行shell命令并且通...
  • liuxingen
  • liuxingen
  • 2015年07月25日 22:06
  • 25162

利用PHP的Popen实现RRDTOOL作图的动态输出

最近看了一下RRDTOOL如何作图,语法确实比较繁琐,但不是太难。命令行下执行的时候,每次都需要输出成为一个文件。我们希望做一个应用,能够动态生成图表,看了看Cacti的实现方法,核心的部分就是使用了...
  • rdstwww
  • rdstwww
  • 2015年10月12日 00:56
  • 875

PHP 调用系统外部命令 system() exec() passthru() 和 popen()

一 用PHP提供的专门函数,简单执行命令 原型: string system (string command [, int return_var]); string exec(string co...
  • wzhwho
  • wzhwho
  • 2011年11月07日 16:57
  • 2487
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:php中popen,exec,system,passthru到底有多大区别
举报原因:
原因补充:

(最多只允许输入30个字)