PHP 命令行工具 shell_exec, exec, passthru, system详细使用介绍


Posted in PHP onSeptember 11, 2011

所有这些命令都衍生一个子进程,用于运行您指定的命令或脚本,并且每个子进程会在命令输出写到标准输出 (stdout) 时捕捉它们。

shell_exec()

shell_exec() 命令行实际上仅是反撇号 (`) 操作符的变体。如果您编写过 shell 或 Perl 脚本,您就知道可以在反撇号操作符内部捕捉其他命令的输出。例如,清单 1 显示了如何使用反撇号在当前目录中获取每个文本(.txt)的单词计数。

清单 1. 使用反撇号计算单词数量

#! /bin/sh 
number_of_words=`wc -w *.txt` 
echo $number_of_words #result would be something like: 
#165 readme.txt 388 results.txt 588 summary.txt 
#and so on....

在您的 PHP 脚本中,您可以在 shell_exec() 中运行这个简单的命令,如清单 2 所示,并获取想要的结果。这里假设在同一个目录下有一些文本文件。

清单 2. 在 shell_exec() 中运行相同的命令

<?php 
$results = shell_exec('wc -w *.txt'); 
echo $results; 
?>

在图 1 中可以看到,获得的结果与从 shell 脚本得到的一样。这是因为 shell_exec() 允许您通过 shell 运行外部程序,然后以字符串的形式返回结果。
图 1. 通过 shell_exec() 运行 shell 命令的结果
PHP 命令行工具 shell_exec, exec, passthru, system详细使用介绍
注意,仅使用后撇号操作符也会得到相同的结果,如下所示。
清单 3. 仅使用后撇号操作符
<?php 
$results = `wc -w *.txt`; 
echo $results; 
?>

清单 4 给出了一种更加简单的方法。
清单 4. 更加简单的方法
<?php 
echo `wc -w *.txt`; 
?>

通过 UNIX 命令行和 shell 脚本能够完成很多东西,知道这点很重要。例如,您可以使用竖线将命令连接起来。您甚至可以使用操作符在其中创建 shell 脚本,并且仅调用 shell 脚本(根据需要使用或不使用参数)。

例如,如果您仅希望计算该目录下的前 5 个文本文件的单词数,那么可以使用竖线 (|) 将 wc 和 head 命令连接起来。另外,您还可以将输出结果放到 pre 标记内部,让它能够更美观地呈现在 Web 浏览器中,如下所示。

清单 5. 更加复杂的 shell 命令

<?php 
$results = shell_exec('wc -w *.txt | head -5'); 
echo "<code lang="php">".$results . "</code>"; 
?>

图 2 演示了运行清单 5 的脚本得到的结果。
图 2. 从 shell_exec() 运行更复杂的 shell 命令得到的结果
PHP 命令行工具 shell_exec, exec, passthru, system详细使用介绍
在本文的后面部分,您将学习如何使用 PHP 为这些脚本传递参数。现在您可以将它看作运行 shell 命令的一种方法,但要记住您只能看到标准输出。如果命令或脚本出现错误,您将看不到标准的错误 (stderr),除非您通过竖线将它添加到 stdout。

passthru()

passthru() 允许您运行外部程序,并在屏幕上显示结果。您不需要使用 echo 或 return 来查看结果;它们会显示在浏览器上。您可以添加可选的参数,即保存从外部程序返回的代码的变量,比如表示成功的 0,这为调试提供更好的机制。

在清单 6 中,我使用 passthru() 命令运行在前面小节运行的单词计数脚本。如您所见,我还添加一个包含返回代码的 $returnval 变量。

清单 6. 使用 passthru() 命令运行单词计数脚本

<?php 
passthru('wc -w *.txt | head -5',$returnval); 
echo "<hr/>".$returnval; 
?>

注意,我不需要使用 echo 返回任何东西。结果会直接显示在屏幕上,如下所示。
图 3. 使用 return 代码运行 passthru() 命令的结果
PHP 命令行工具 shell_exec, exec, passthru, system详细使用介绍
在清单 7 中,我通过删除脚本头部的 5 前面的短横线 (-) 引入一个小错误。
清单 7. 在单词计数脚本中引入一个错误
<?php 
//we introduce an error below (removing - from the head command) passthru('wc -w *.txt | head 5',$returnval); 
echo "<hr/>".$returnval; 
?>

注意,脚本未能按照预期运行。您得到的是一个空白的屏幕,一条水平线和返回值 1,如图 4 所示。这个返回代码通常表明发生了某些错误。如果能够测试返回代码,查找和修复错误就容易多了。
图 4. 使用 passthru() 时查看错误代码
PHP 命令行工具 shell_exec, exec, passthru, system详细使用介绍
exec()
exec() 命令与 shell_exec() 相似,不同之处是它返回输出的最后一行,并且可选地用命令的完整输出和错误代码填充数组。清单 8 展示了当运行 exec() 而不捕捉数据数组中的数据时发生的事情。
清单 8. 运行 exec() 而不捕捉数据数组中的数据
<?php 
$results = exec('wc -w *.txt | head -5'); 
echo $results; 
#would print out just the last line or results, i.e.: 
#3847 myfile.txt 
?>

为了捕捉数组中的结果,要将该数组的名称作为第二个参数添加到 exec()。我在清单 9 中执行了这个步骤,并以 $data 作为数组的名称。
清单 9. 从 exec() 捕捉数据数组的结果
<?php 
$results = exec('wc -w *.txt | head -5',$data); 
print_r($data); 
#would print out the data array: 
#Array ( [0]=> 555 text1.txt [1] => 283 text2.txt) 
?>

在捕捉数组中的结果之后,您可以对每行进行一些处理。例如,您可以在第一个空格处进行划分,将分离的值存储在数据库表中,或对每个行应用特定的格式或标记。
system()
如清单 10 所示,system() 命令是一种混合体。它像 passthru() 一样直接输出从外部程序接收到的任何东西。它还像 exec() 一样返回最后一行,并使返回代码可用。
清单 10. system() 命令
<?php 
system('wc -w *.txt | head -5'); 
#would print out: 
#123 file1.txt 332 file2.txt 444 file3.txt 
#and so on 
?>

一些例子
现在您已经了解如何使用这些 PHP 命令,但可能仍然有一些疑问。例如,什么时候应该使用哪个命令?这完全由您的需求决定。
大多数情况下,我使用 exec() 命令和数据数组处理所有东西。或者对更简单的命令使用 shell_exec(),尤其是不关心结果时。如果仅需返回一个 shell 脚本,我就使用 passthru()。通常,我在不同的场合中使用不同的函数,并且有时它们是可以互换的。这完全取决于我的心情和要实现的目的。
您可能提问的另一个问题是 “它们的长处是什么?”。如果您没有头绪,或者一个项目非常适合使用 shell 命令,但不知道如何使用,那么我在这里提供一些见解。
如果您正在编写一个提供各种备份或文件传输功能的应用程序,您可以选择使用 shell_exec() 或这里提供的其他命令之一运行 rsync 支持的 shell 脚本。您可以编写 shell 脚本使其包含必要的 rsync 命令,然后使用 passthru() 根据用户的命令或 cron 作业执行它。
例如,一位用户在您的应用程序中有适当的权限(比如管理员权限),他想将 50 个 PDF 文件从一个服务器发送到另一个服务器。那么,该用户需要在应用程序中导航到正确的位置,单击 Transfer,选择需要发送的 PDF,然后单击 Submit。在这个过程中,该表单应该有一个 PHP 脚本,它使用返回选项变量通过 passthru() 运行 rsync 脚本,这样您就知道是否发生问题,如下所示。
清单 11. 通过 passthru() 运行 rsync 脚本的示例 PHP 脚本
<?php 
passthru('xfer_rsync.sh',$returnvalue); 
if ($returnvalue != 0){ 
//we have a problem! 
//add error code here 
}else{ 
//we are okay 
//redirect to some other page 
} 
?>

如果您的应用程序需要列出进程或文件,或关于这些进程或文件的数据,您可以使用本文总结的命令之一轻松实现这个目的。例如,一个简单的 grep 命令能够帮助您找到匹配特定搜索条件的文件。将它与 exec() 命令一起使用可以将结果保存到一个数组中,这允许您构建一个 HTML 表或表单,它们又进一步允许您运行其他命令。
到目前为止,我讨论了用户生成的事件 —— 用户只要按下按钮或单击链接,PHP 就运行相应的脚本。您还可以将独立的 PHP 脚本和 cron 或其他日程安排程序一起使用,从而实现一些有趣的效果。例如,如果您一个备份脚本,您可以通过 cron 运行它,或者将它打包到 PHP 脚本后在运行。为什么要这样做?这似乎是多余的,不是吗?不是这样的 —— 您需要这样考虑,您可以通过 exec() 或 passthru() 运行备份脚本,然后根据返回代码执行一些行为。如果出现错误,您可以将其记录到错误日志或数据库中,或发送一封警告电子邮件。如果脚本成功,您可以将原始的输出转储到数据库(例如,rsync 有一个详尽(verbose)模式,对随后诊断问题十分有用)。

--------------------------------------------------------------------------------
安全
我们在这里简要讨论一下安全性:如果您接受用户输入并将信息传递到 shell,那么最好过滤用户输入。删除您认为有害的命令和不允许的内容,比如 sudo(作为超级用户运行)或 rm(删除)。事实上,您可能不希望用户发送开放的请求,而是让他们从列表中选择。
例如,您运行一个接受文件列表作为参数的传输程序,您应该通过一系列复选框列出所有文件。用户可以选择和取消选择文件,并通过单击 Submit 激活 rsync shell 脚本。用户不能自己输入文件或使用正则表达式。

--------------------------------------------------------------------------------
结束语
在本文中,我演示了使用 PHP 命令运行 shell 脚本和其他命令的基础知识。这些 PHP 命令包括 shell_exec()、exec()、passthru() 和 system()。现在,您应该在自己的应用程序中实践学到的知识。

PHP 相关文章推荐
一些被忽视的PHP函数(简单整理)
Apr 30 PHP
浅析get与post的一些特殊情况
Jul 28 PHP
php中通过DirectoryIterator删除整个目录的方法
Mar 13 PHP
mac系统下为 php 添加 pcntl 扩展
Aug 28 PHP
php版微信数据统计接口用法示例
Oct 12 PHP
PHP微信分享开发详解
Jan 14 PHP
利用laravel+ajax实现文件上传功能方法示例
Aug 13 PHP
PHPMailer使用QQ邮箱实现邮件发送功能
Aug 18 PHP
PHP开发之归档格式phar文件概念与用法详解【创建,使用,解包还原提取】
Nov 17 PHP
PhpStorm本地断点调试的方法步骤
May 21 PHP
PHP JWT初识及其简单示例
Oct 10 PHP
Yii2框架配置文件(Application属性)与调试技巧实例分析
May 27 PHP
20个PHP常用类库小结
Sep 11 #PHP
php各种编码集详解和以及在什么情况下进行使用
Sep 11 #PHP
php正则表达式(regar expression)
Sep 10 #PHP
PHP setcookie指定domain参数后,在IE下设置cookie失效的解决方法
Sep 09 #PHP
判断PHP数组是否为空的代码
Sep 08 #PHP
PHP中通过语义URL防止网站被攻击的方法分享
Sep 08 #PHP
PHP session会话的安全性分析
Sep 08 #PHP
You might like
表单提交验证类
2006/07/14 Javascript
JScript内置对象Array中元素的删除方法
2007/03/08 Javascript
让 JavaScript 轻松支持函数重载 (Part 2 - 实现)
2009/08/04 Javascript
javascript 系统文件夹文件操作及参数介绍
2013/01/08 Javascript
关于JavaScript中string 的replace
2013/04/12 Javascript
使用js画图之圆、弧、扇形
2015/01/12 Javascript
jquery实现点击其他区域时隐藏下拉div和遮罩层的方法
2015/12/23 Javascript
浅析script标签中的defer与async属性
2016/11/30 Javascript
微信小程序 开发之快递查询功能的实现
2017/01/09 Javascript
Javascript的this用法
2017/01/16 Javascript
angularjs点击图片放大实现上传图片预览
2017/02/24 Javascript
js实现点击切换checkbox背景图片的简单实例
2017/05/08 Javascript
详解Javascript获取缓存和清除缓存API
2017/05/25 Javascript
layUI的验证码功能及校验实例
2019/10/25 Javascript
JavaScript实现Excel表格效果
2020/02/07 Javascript
如何封装Vue Element的table表格组件
2021/02/06 Vue.js
[38:39]KG vs Mineski 2019国际邀请赛小组赛 BO2 第一场 8.15
2019/08/16 DOTA
Python实战小程序利用matplotlib模块画图代码分享
2017/12/09 Python
Python3.6笔记之将程序运行结果输出到文件的方法
2018/04/22 Python
Python3日期与时间戳转换的几种方法详解
2019/06/04 Python
带你彻底搞懂python操作mysql数据库(cursor游标讲解)
2020/01/06 Python
Python使用进程Process模块管理资源
2020/03/05 Python
使用python实现CGI环境搭建过程解析
2020/04/28 Python
Python多线程的退出控制实现
2020/08/10 Python
Python私有属性私有方法应用实例解析
2020/09/15 Python
Selenium结合BeautifulSoup4编写简单的python爬虫
2020/11/06 Python
解决python3中os.popen()出错的问题
2020/11/19 Python
微软巴西官方网站:Microsoft Brasil
2019/09/26 全球购物
网络工程师个人的自我评价范文
2013/10/01 职场文书
师范生求职自荐信
2014/06/14 职场文书
向国旗敬礼活动总结范文2014
2014/09/27 职场文书
房屋授权委托书范本
2014/10/07 职场文书
2014年流动人口工作总结
2014/11/26 职场文书
2015会计试用期工作总结
2014/12/12 职场文书
幼儿园班级工作总结2015
2015/05/25 职场文书
导游词之烟台威海蓬莱
2019/11/14 职场文书