时间:2021-07-01 10:21:17 帮助过:88人阅读
在php代码中需要执行一个shell脚本,希望分别得到它的标准输出(stdout)和错误输出(stderr)。用shell的重定向把标准输出和错误输出放到文件中再用php读取文件,可行,但输出文件的内容传回php后就不再需要,有没有办法把shell的输出重定向到变量?
要执行脚本的功能,无法用php本身完成(执行R语言的脚本),并且需要把常规输出和错误输出都反馈。
shell里面的重定向可以把标准输出和错误输出分别放到文件中,但产生临时文件。
my_cmd 1>out.txt 2>err.txt
php再读取out.txt
和err.txt
文件内容进行返回。
这样做的缺点:产生了中间文件。如果shell命令my_cmd
执行的时间比较长(可能要几分钟),那么同一用户如果短时间内多次发起http请求使得php执行my_cmd
,则需要每次out.txt
和err.txt
的名字都不同,以免写入冲突。这就产生了很多不必要的中间文件。即使及时删除它们,对于I/O还是有开销,希望能避免。
php的exec()
函数:
string exec ( string $command [, array &$output [, int &$return_var ]] )
里面的参数$output
看起来只能得到“运行脚本时屏幕输出”的内容,是stdout或者stdout和stderr的混合,不能分开获得。
在php代码中需要执行一个shell脚本,希望分别得到它的标准输出(stdout)和错误输出(stderr)。用shell的重定向把标准输出和错误输出放到文件中再用php读取文件,可行,但输出文件的内容传回php后就不再需要,有没有办法把shell的输出重定向到变量?
要执行脚本的功能,无法用php本身完成(执行R语言的脚本),并且需要把常规输出和错误输出都反馈。
shell里面的重定向可以把标准输出和错误输出分别放到文件中,但产生临时文件。
my_cmd 1>out.txt 2>err.txt
php再读取out.txt
和err.txt
文件内容进行返回。
这样做的缺点:产生了中间文件。如果shell命令my_cmd
执行的时间比较长(可能要几分钟),那么同一用户如果短时间内多次发起http请求使得php执行my_cmd
,则需要每次out.txt
和err.txt
的名字都不同,以免写入冲突。这就产生了很多不必要的中间文件。即使及时删除它们,对于I/O还是有开销,希望能避免。
php的exec()
函数:
string exec ( string $command [, array &$output [, int &$return_var ]] )
里面的参数$output
看起来只能得到“运行脚本时屏幕输出”的内容,是stdout或者stdout和stderr的混合,不能分开获得。
找到答案了,自问自答一次。
使用proc_open()
函数,它能执行一段shell脚本,并把stdout和stderr分别保存,还支持设定stdin:
resource proc_open ( string $cmd , array $descriptorspec , array &$pipes [, string $cwd [, array $env [, array $other_options ]]] )
参考代码:
$descriptorspec = array(
0 => array("pipe", "r"), // stdin
1 => array("pipe", "w"), // stdout
2 => array("pipe", "w") // stderr
);
$cmd = 'Rscript hello.r'; // 替换为你要执行的shell脚本
$proc = proc_open($cmd, $descriptorspec, $pipes, null, null);
// $proc为false,表明命令执行失败
if ($proc == false) {
// do sth with HTTP response
} else {
$stdout = stream_get_contents($pipes[1]);
fclose($pipes[1]);
$stderr = stream_get_contents($pipes[2]);
fclose($pipes[2]);
$status = proc_close($proc); // 释放proc
}
$data = array(
'code' => 0,
'msg' => array(
'stdout' => $stdout,
'stderr' => $stderr,
'retval' => $status
)
);
// do sth with $data to HTTP response