PHP 函数proc_open笔记

双管道,多线程执行命令: 使用示例:

test.php:

$descriptorspec=array(
    0=>STDIN,
    1=>STDOUT,
    2=>STDERR
);

$process=proc_open('mysql -u root -p',$descriptorspec,$pipes)

这样 运行php test.php 看见的效果将和终端直接运行mysql命令一模一样。

$descriptorspec说明, 它表述了执行命令的执行文件获得从哪儿获得STDIN以及STDOUT输出到哪儿, STDERR输出到哪儿。

他一般有三种形式, 资源,管道,文件

管道表示:

$descriptorspec=array(
    0=>array('pipe','r'),
    1=>array('pipe','w'),
    2=>array('pipe','w'),
);

管道操作:

if(is_resource($process)){
    fwrite($pipes[0],'content');

    while($ret=fgets[$pipes[1]){
       echo ''.$ret;
    }
    while($ret=fgets[$pipes[2]){
    echo ''.$ret;
}
}
注意事项:

1,读取管道时,有时候不能用stream_get_contents获得内容,必须用fgets, 这时候往往是因为命令执行文件有while死循环,不能结束进程,stream_get_contents是程序结束进程后才能

读取到。 而fgets 每次只读取一行, 要求命令执行文件输出内容需要带换行。 并且有时候需要用while语句循环来读取, 因为fgets每次只读一行 这样保证所有的都读取完了。

2,再读取管道时,往往会堵塞。但是不影响下面程序的执行。 如:

$ret=fgets($pipes[1]);//这里有堵塞。
echo $ret;
fwrite($pipes[0],'content');//但是这里可以执行。

这时候fgets 不能加while循环。


3,stream_set_blocking函数可以设置是否堵塞,但是不知道为什么对管道好像不起作用。

4,将读取内容加上空字符串后才进行echo输出的目的, 是将STDERR, STDOUT等特殊类型的字符串转换为标准字符串 ,不然有些程序将读取不了这些字符串,比如用  ob_get_clean() 有可能读取不了。  

public function __construct($cmd){
        $this->pipes = null;
        $this->descriptors = array(
            0 => array('pipe', 'r'),
            1 => array('pipe', 'w'),
            2 => array('pipe', 'w')
        );
        $this->resource = proc_open($cmd, $this->descriptors, $this->pipes);
        stream_set_blocking($this->pipes[0], FALSE);
        stream_set_blocking($this->pipes[1], FALSE);
    }

$status = proc_get_status($this->resource);
Array
(
    [command] => top
    [pid] => 7985
    [running] => 1
    [signaled] =>
    [stopped] =>
    [exitcode] => -1
    [termsig] => 0
    [stopsig] => 0
)

switch($status['exitcode']) {
     case '255':
     case '-1':
     case '1':
         return false;
     case '0':
         // normal end
         return false;
     default:
         // unknown
}

public function errors(){
     $msg = 'Error:';
     while (is_resource($this->pipes[self::PIPE_STDOUT]) && !feof($this->pipes[self::PIPE_STDOUT])) {
        $msg .= fgets($this->pipes[self::PIPE_STDOUT]);
     }
     while (is_resource($this->pipes[self::PIPE_STDERR]) && !feof($this->pipes[self::PIPE_STDERR])) {
        $msg .= fgets($this->pipes[self::PIPE_STDERR]);
     }
     fclose($this->pipes[self::PIPE_STDERR]);
     proc_close($this->resource);
     return $msg;
}



  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
proc_open()是PHP的一个函数,用于创建一个新的进程并与它进行通信。以下是一个示例: ``` $descriptorspec = array( 0 => array("pipe", "r"), // 标准输入 1 => array("pipe", "w"), // 标准输出 2 => array("pipe", "w") // 标准错误输出 ); $process = proc_open('ls -la', $descriptorspec, $pipes, '/tmp'); if (is_resource($process)) { fwrite($pipes[0], "input\n"); fclose($pipes[0]); echo stream_get_contents($pipes[1]); fclose($pipes[1]); echo stream_get_contents($pipes[2]); fclose($pipes[2]); $returnValue = proc_close($process); } ``` 该示例中,我们运行了一个shell命令`ls -la`,并在`/tmp`目录下执行。`$descriptorspec`是一个数组,指定了标准输入、标准输出和标准错误输出的管道。 我们使用`proc_open()`函数创建了一个新的进程。如果此函数成功执行,它将返回一个进程资源。我们可以使用`is_resource()`函数来检查此进程资源是否有效。 接下来,我们向标准输入管道中写入了一个字符串`input`,并通过`fclose()`关闭了该管道。然后我们使用`stream_get_contents()`函数读取了标准输出和标准错误输出的管道,并通过`fclose()`函数关闭了它们。 最后,我们使用`proc_close()`函数关闭进程并获取该进程的退出代码。 请注意:使用`proc_open()`函数时需要谨慎,因为它允许执行任意的shell命令。为了避免安全问题,请确保在执行时正确地过滤和验证输入。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值