从各种互联网帖子和最近的经验来看,我发现你不能依赖 proc_close 返回子进程的准确返回代码。 返回代码还取决于你是否从 stdout/stderr 管道读取,就像我的示例所示。 我通过将退出代码写入额外的文件描述符来解决这个问题。
<?
$descriptorspec = array(
0 => array('pipe', 'r'), // stdin 是子进程将从中读取的管道
1 => array('pipe', 'w'), // stdout 是子进程将写入的管道
2 => array('pipe', 'w'), // stderr 是子进程将写入的管道
);
$proc = @proc_open("/bin/ls -l /etc/passwd", $descriptorspec, $pipes);
fclose($pipes[0]);
$output = array();
while (!feof($pipes[1])) array_push($output, rtrim(fgets($pipes[1],1024),"\n"));
fclose($pipes[1]);
while (!feof($pipes[2])) array_push($output, rtrim(fgets($pipes[2],1024),"\n"));
fclose($pipes[2]);
$exit=proc_close($proc);
print_r($output);
echo "exitcode $exit\n\n";
$descriptorspec = array(
0 => array('pipe', 'r'), // stdin 是子进程将从中读取的管道
1 => array('pipe', 'w'), // stdout 是子进程将写入的管道
2 => array('pipe', 'w'), // stderr 是子进程将写入的管道
);
$proc = @proc_open("/bin/ls -l /etc/passwd", $descriptorspec, $pipes);
fclose($pipes[0]);
fclose($pipes[1]);
fclose($pipes[2]);
$exit=proc_close($proc);
echo "exitcode $exit\n\n";
$descriptorspec = array(
0 => array('pipe', 'r'), // stdin 是子进程将从中读取的管道
1 => array('pipe', 'w'), // stdout 是子进程将写入的管道
2 => array('pipe', 'w'), // stderr 是子进程将写入的管道
3 => array('pipe', 'w'), // stderr 是子进程将写入的管道
);
$proc = @proc_open("/bin/ls -l /etc/passwd;echo $? >&3", $descriptorspec, $pipes);
fclose($pipes[0]);
$output = array();
// 注释下一行以获取正确的 exicode
while (!feof($pipes[1])) array_push($output, rtrim(fgets($pipes[1],1024),"\n"));
fclose($pipes[1]);
while (!feof($pipes[2])) array_push($output, rtrim(fgets($pipes[2],1024),"\n"));
fclose($pipes[2]);
if (!feof($pipes[3])) $output['exitcode']=rtrim(fgets($pipes[3],5),"\n");
fclose($pipes[3]);
proc_close($proc);
print_r($output);
?>
在我的系统上输出
数组
(
[0] => -rw-r--r-- 1 root root 1460 2005-09-02 09:52 /etc/passwd
[1] =>
[2] =>
)
exitcode -1
exitcode 1
数组
(
[0] => -rw-r--r-- 1 root root 1460 2005-09-02 09:52 /etc/passwd
[1] =>
[2] =>
[exitcode] => 0
)