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学习 计数器实例代码
Jun 15 PHP
php 无限级数据JSON格式及JS解析
Jul 17 PHP
PHP $_FILES函数详解
Mar 09 PHP
深入解析php中的foreach问题
Jun 30 PHP
非常好用的Zend Framework分页类
Jun 25 PHP
php实现读取超大文件的方法
Jul 28 PHP
PHP连接和操作MySQL数据库基础教程
Sep 29 PHP
详解PHP中的序列化、反序列化操作
Mar 21 PHP
thinkphp自定义权限管理之名称判断方法
Apr 01 PHP
Yii框架日志记录Logging操作示例
Jul 12 PHP
Laravel 集成微信用户登录和绑定的实现
Dec 27 PHP
实例化php类时传参的方法分析
Jun 05 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
WINDOWS下php5.2.4+mysql6.0+apache2.2.4+ZendOptimizer-3.3.0配置
2008/03/28 PHP
php表单加入Token防止重复提交的方法分析
2016/10/10 PHP
Yii2框架实现数据库常用操作总结
2017/02/08 PHP
YII2框架中使用yii.js实现的post请求
2017/04/09 PHP
form自动提交实例讲解
2017/07/10 PHP
thinkPHP通用控制器实现方法示例
2017/11/23 PHP
PHP快速导出百万级数据到CSV或者EXCEL文件
2020/11/27 PHP
ASP.NET中基于JQUERY的高性能的TreeView补充
2011/02/23 Javascript
jQuery对表单元素的取值和赋值操作代码
2011/05/19 Javascript
JavaScript(js)设置默认输入焦点(focus)
2012/12/28 Javascript
javascript判断非数字的简单例子
2013/07/18 Javascript
js showModalDialog参数的使用详解
2014/01/07 Javascript
jQuery学习笔记之jQuery.fn.init()的参数分析
2014/06/09 Javascript
js获取当前年月日-YYYYmmDD格式的实现代码
2016/06/01 Javascript
JS弹出新窗口被拦截的解决方法
2016/08/09 Javascript
ReactNative 之FlatList使用及踩坑封装总结
2017/11/29 Javascript
在vue中使用公共过滤器filter的方法
2018/06/26 Javascript
JavaScript实现猜数字游戏
2020/05/20 Javascript
VueJS实现用户管理系统
2020/05/29 Javascript
如何利用nodejs实现命令行游戏
2020/11/24 NodeJs
[53:10]Secret vs Pain 2018国际邀请赛小组赛BO2 第一场 8.17
2018/08/20 DOTA
Python读写Json涉及到中文的处理方法
2016/09/12 Python
Python 装饰器深入理解
2017/03/16 Python
Python面向对象特殊成员
2017/04/24 Python
import的本质解析
2017/10/30 Python
Python实现的直接插入排序算法示例
2018/04/29 Python
python基于opencv 实现图像时钟
2021/01/04 Python
巧用CSS3的calc()宽度计算做响应模式布局的方法
2018/03/22 HTML / CSS
HTML5 canvas画矩形时出现边框样式不一致的解决方法
2013/10/14 HTML / CSS
Camper鞋西班牙官方网上商店:西班牙马略卡岛的鞋类品牌
2019/03/14 全球购物
美国精油公司:Plant Therapy
2019/05/17 全球购物
MySQL中一条update语句是如何执行的
2022/03/16 MySQL
Spring Security使用单点登录的权限功能
2022/04/03 Java/Android
Python字符串格式化方式
2022/04/07 Python
Python加密与解密模块hashlib与hmac
2022/06/05 Python
oracle设置密码复杂度及设置超时退出的功能
2022/06/28 Oracle