我会使用以下代码:
$SERVER_PATH = dirname(__FILE__);
shell_exec($PHP_LOCATION.' '.$SERVER_PATH."/script.php?k1=v1&k2=v2 > /dev/null 2>/dev/null &");
地点:
$PHP_LOCATION
应该包含 PHP 的路径,$SERVER_PATH
- 是当前工作目录(幸运的是要运行的脚本在同一目录中),>/dev/null 2>/dev/null &
添加以使此调用异步(取自 Asynchronous shell exec in PHP 问题)
这段代码有两个问题:
据我所知,
?k1=v1&k2=v2
仅适用于网络调用,因此在这种特殊情况下,参数不会传递给脚本。我真的不知道如何初始化
$PHP_LOCATION
变量以灵活地在大多数主机上工作。
我对这两个问题进行了一些研究:
解决1 suggested使用 -- 'parameters_string'
但也建议修改脚本以解析看起来有点笨拙的参数字符串。有更好的解决方案吗?
为了解决2,我找到了一个solution使用 PHP_BINARY
但这是 PHP 5.4+ 案例(我使用的是 5.3)。但是原来的问题是要运行与原始脚本版本相同版本的PHP。那么对我来说(因为我只使用 PHP 5.3)可能有解决方案吗?
编辑 0
让我解释一下为什么我坚持使用这种奇怪的(对于 PHP)方法:
那些 PHP 脚本应该彼此分开:
其中一个将分析数据并
第二个将生成 PNG 图作为最终结果。
这些脚本并非旨在同时运行,这意味着第二个脚本可以按照自己的时间表运行,只需要在其数据准备好(由第一个脚本完成)时运行。因此不应将数据从第二个脚本(子)传递回第一个(父)。
编辑 1
从大多数评论来看,主要讨论都朝着 fork 方向发展。但是,我想强调一下原始问题中提出的 1 和 2 点。我有一些理由按照我指出的方式解决任务,我试图指出所有这些理由。如果我的某些观点看起来很奇怪,请发表评论 - 我会把它说得更清楚,否则我会更改主要问题。
提前致谢!
最佳答案
如何获取可执行文件
假设您使用的是 Linux,您可以使用:
function getBinaryRunner($binary)
{
return trim(shell_exec('which '.$binary));
}
例如,同样可以用来检查是否安装了需要的东西:
function checkIfCommandExists($command)
{
$result = shell_exec('which '.$command);
return !empty($result);
}
几点:
- 是的,它只适用于 Linux
- 如果允许将用户输入传递给 shell 命令,您应该小心用户输入:
escapeshellarg()
和公司 - 事实上,通常 PHP 不应该用于此类事情,就好像它是关于异步请求的,最好是实现 fork 或从外部工作人员运行命令。
如何传递参数
通过文件系统路径执行 shell_exec()
您正在访问文件,显然,所有“GET”参数都只是文件名的一部分,它不再是“URI”没有网络服务器来处理它。所以你有两个选择:
通过访问您的网络服务器调用调用。所以它会像:
//Yes, you will use wget or, better, curl to make web-request from CLI shell_exec('wget http://your.web-server.domain/script.php?foo=bar');
这里的缺点:如果您将通过公共(public) DNS 访问您的网络服务器,它将导致网络中断和所有处理开销。好处 - 显然,您不必在脚本中期望任何其他内容,并且不区分 CLI 和非 CLI 调用
使用
$_SERVER
数组中的数组并传递 CLI 应有的参数:shell_exec('/usr/bin/php /path/to/script.php foo bar'); //inside your script.php you will see: //$_SERVER['argv'][0] is "script.php" //$_SERVER['argv'][1] is "foo" //$_SERVER['argv'][2] is "bar"
是的,这将需要修改脚本,并且可能需要一些关于如何映射“常规”网络请求和 CLI 请求的逻辑。我什至建议考虑将与 CLI 相关的内容分离到不同的脚本包中,以免弄乱该逻辑。
关于“异步运行”的更多信息
当您执行 php script.php &
时,您只是在后台模式下运行它。但是,这仍然会为您的流程保持父子关系。这意味着 - 如果父进程死亡,它的子进程也将被删除。准确地说,SIGHUP
将被触发,为避免这种情况,您应该使用 nohup
命令。它将允许模拟进程的“分离”,因此使其运行可靠且独立于父进程发生的情况。
关于php - 从 PHP 脚本运行 PHP 脚本的最佳实践是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31558523/