在php中执行程序 - 显示并返回输出

时间:2012-02-27 16:02:57

标签: php shell

在php中有几种执行shell命令的方法:

  • 系统()
  • 中继()
  • 了shell_exec()
  • EXEC()

前两个显示输出但不返回。 最后两个返回输出但不显示它。

我想运行shell命令,这需要很多时间,但它会显示一些输出,所以我知道它不会挂起。但最后我想在php中处理这个输出。如果我选择前两个中的一个,我将无法获得输出,因此我将无法在php中处理它。如果我运行最后两个中的一个,我将能够处理输出但是我的程序将挂起很长时间而不输出任何东西。

有没有办法运行一个shell命令,它会立即显示输出并返回它?

2 个答案:

答案 0 :(得分:2)

也许你可以使用popen()执行一个程序并通过文件句柄读取它的输出,如下所示:

$handle = popen('/bin/ls', 'r');
while ($line = fread($handle, 100)){
    echo $line;
}
pclose($handle);

答案 1 :(得分:2)

也许这个人会对你感兴趣吗? proc_open() - http://www.php.net/manual/en/function.proc-open.php

这是一个方便的代码片段,可能适合你(它是从我给你链接的网站上的评论中复制的):

<?php
/*
 * Execute and display the output in real time (stdout + stderr).
 *
 * Please note this snippet is prepended with an appropriate shebang for the 
 * CLI. You can re-use only the function.
 * 
 * Usage example:
 * chmod u+x proc_open.php
 * ./proc_open.php "ping -c 5 google.fr"; echo RetVal=$?
 */
define(BUF_SIZ, 1024);        # max buffer size
define(FD_WRITE, 0);        # stdin
define(FD_READ, 1);        # stdout
define(FD_ERR, 2);        # stderr

/*
 * Wrapper for proc_*() functions.
 * The first parameter $cmd is the command line to execute.
 * Return the exit code of the process.
 */
function proc_exec($cmd)
{
    $descriptorspec = array(
        0 => array("pipe", "r"),
        1 => array("pipe", "w"),
        2 => array("pipe", "w")
    );

    $ptr = proc_open($cmd, $descriptorspec, $pipes, NULL, $_ENV);
    if (!is_resource($ptr))
        return false;

    while (($buffer = fgets($pipes[FD_READ], BUF_SIZ)) != NULL 
            || ($errbuf = fgets($pipes[FD_ERR], BUF_SIZ)) != NULL) {
        if (!isset($flag)) {
            $pstatus = proc_get_status($ptr);
            $first_exitcode = $pstatus["exitcode"];
            $flag = true;
        }
        if (strlen($buffer))
            echo $buffer;
        if (strlen($errbuf))
            echo "ERR: " . $errbuf;
    }

    foreach ($pipes as $pipe)
        fclose($pipe);

    /* Get the expected *exit* code to return the value */
    $pstatus = proc_get_status($ptr);
    if (!strlen($pstatus["exitcode"]) || $pstatus["running"]) {
        /* we can trust the retval of proc_close() */
        if ($pstatus["running"])
            proc_terminate($ptr);
        $ret = proc_close($ptr);
    } else {
        if ((($first_exitcode + 256) % 256) == 255 
                && (($pstatus["exitcode"] + 256) % 256) != 255)
            $ret = $pstatus["exitcode"];
        elseif (!strlen($first_exitcode))
            $ret = $pstatus["exitcode"];
        elseif ((($first_exitcode + 256) % 256) != 255)
            $ret = $first_exitcode;
        else
            $ret = 0; /* we "deduce" an EXIT_SUCCESS ;) */
        proc_close($ptr);
    }

    return ($ret + 256) % 256;
}

/* __init__ */
if (isset($argv) && count($argv) > 1 && !empty($argv[1])) {
    if (($ret = proc_exec($argv[1])) === false)
        die("Error: not enough FD or out of memory.\n");
    elseif ($ret == 127)
        die("Command not found (returned by sh).\n");
    else
        exit($ret);
}
?>