美文网首页
PHP进程间通信--有名管道

PHP进程间通信--有名管道

作者: 大胡子商人 | 来源:发表于2017-11-07 10:04 被阅读180次

管道PIPE

管道用于承载简称之间的通讯数据。为了方便理解,可以将管道比作文件,进程A将数据写到管道P中,然后进程B从管道P中读取数据。php提供的管道操作API与操作文件的API基本一样,除了创建管道使用posix_mkfifo函数,读写等操作均与文件操作函数相同。当然,你可以直接使用文件模拟管道,但是那样无法使用管道的特性了。

通过管道通信的大概思路是,首先创建一个管道,然后子进程向管道中写入信息,父进程从管道中读取信息,这样就可以做到父子进程直接实现通信了。

代码如下:

<?php
   /**
    * author: NickBai
    * createTime: 2016/12/2 0002 上午 11:12
    */
   //创建管道
   $pipePath = "/tmp/test.pipe";
    if( !file_exists( $pipePath ) ){
      if( !posix_mkfifo( $pipePath, 0666 ) ){
         exit('make pipe false!' . PHP_EOL);
    }
}
 
//创建进程,子进程写管道,父进程读管道
$pid = pcntl_fork();

if( $pid == 0 ){
     //子进程写管道
     $file = fopen( $pipePath, 'w' );
     fwrite( $file, 'hello world' );
     sleep(1);
     exit();
}else{
     //父进程读管道
     $file = fopen( $pipePath, 'r' );
     //stream_set_blocking( $file, False );  //设置成读取非阻塞
     echo fread( $file, 20 ) . PHP_EOL;
 
     pcntl_wait($status);  //回收子进程
}

注意:本代码只能在linux下运行,并且只能在php-cli模式下。
第7行:指定一个管道的路径,这里跟普通文件没什么区别。
第9行:通过 posix_mkfifo 函数创建 管道 并且设置读写权限为 0666
第15行:通过 pcntl_fork函数创建一个子进程。注意从现在开始,程序将会被分成两个进程来执行。 pcntl_fork 函数 很特殊,它调用一次拥有 多个返回值。在父进程中:它返回 子进程的ID 这个值是 大于0 的。在子进程中,它返回0。当返回 -1 时表示创建进程失败。
第17行:两个进程根据当前进程所获得的$pid的值不同,而进入不同的分支。
第18~22行:子进程打开管道,并向其中写入hello world ,然后进入休眠,休眠结束之后,退出。
第25~29行:父进程打开管道,并进行读取,最后执行 29行的代码回收掉子进程。这里面两个地方是阻塞的,首先是默认读的地方,要等待子进程发出exit命令之后,才能返回数据。还有就是回收进程的 pcntl_wait方法。要等到进程退出。

在linux 下运行该代码:


image.png

会看到程序阻塞 1秒 之后,输出 hello world。
当我们打开 第 26 行代码,并将 27行改为 var_dump(fread( $file, 20 )) . PHP_EOL; 时,运行程序:


image.png

能看到程序立马输出 空串,并等待 1秒 中之后退出。这是因为。当读取是非阻塞的情况下,父进程进行读取信息的时候,不会等待立马有信息,管道中没有信息,也会立马返回。然后执行到 29行回收子进程的时候,阻塞等待子进程退出后结束。
下面来看一个简单的实际小例子。两个子进程向一个文件中写信息,父进程负责监听检测这个文件是否写入完成,完成之后,讲这个文件copy一份。这里,父子进程之间通过管道通信,确认是否完成写入。

<?php
/**
 * author: NickBai
 * createTime: 2016/12/2 0002 下午 2:00
 */
//创建管道
$pipePath = "/tmp/test.pipe";
if( !file_exists( $pipePath ) ){
    if( !posix_mkfifo( $pipePath, 0666 ) ){
        exit("make pipe fail \n");
    }
}
 
//创建两个子进程写文件
for( $i = 0; $i < 2; $i++ ){

    $pid = pcntl_fork();
    if( $pid == 0 ){
        file_put_contents( './pipe.log', $i . " write pipe\n", FILE_APPEND );  //写入文件
        $file = fopen( $pipePath, 'w' );
        fwrite( $file, $i . "\n" );  //向管道中写标识,标识写入完毕。
        fclose( $file );
        exit();  //退出子进程
    }
}
 
//父进程要做的是:
//1、读取管道中的写出状态,判断是否完全写完
//2、拷贝写好的文件
//3、删除管道
//4、回收进程
 
$file = fopen( $pipePath, 'r' );
$line = 0;
while(1){
    $end = fread( $file, 1024 );
    foreach( str_split( $end ) as $c) {
         if ( "\n" == $c ) {
             $line++;
         }
    }
 
    if( $line == 2 ){
        copy( './pipe.log', './pipe_copy.log' );
        fclose( $file );
        unlink( $pipePath );
        pcntl_wait( $status );
        exit("ok \n");
    }
}

相关文章

  • PHP进程间通信--有名管道

    管道PIPE 管道用于承载简称之间的通讯数据。为了方便理解,可以将管道比作文件,进程A将数据写到管道P中,然后进程...

  • Linux进程间通信

    Linux进程间通信的概念 linux下进程间通信的几种主要手段简介: 管道(Pipe)及有名管道(named p...

  • 进程间通信方式

    管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字...

  • 进程通信方式

    1、管道与有名管道管道可用于具有亲缘关系的父子进程间通信,有名管道除了具有管道的特点外,还允许无亲缘关系的进程进行...

  • PHP进程间通信--消息队列

    前面介绍了怎么通过消息管道(有名消息管道)进行进程间通信,下面介绍如何通过消息队列实现进程间通信。 首先我们来看一...

  • 系统编程(4)

    哲学家就餐问题: 进程间通信 无名管道,需要亲子进程来实现 利用有名管道产生不同窗口的本地聊天 共享内存间通信,使...

  • LInux进程之间的通信-有名管道(FIFO)

    Linux进程间的通信-有名管道FIFO 管道的通信只能存在于具有亲缘关系的进程之间,比如fork出来的子进程与父...

  • 14.进程间通信:管道

    1. 管道,有名管道和无名管道。 1.1 无名管道主要用于父子进程或者兄弟关系的进程间的通信。通过pipe创建无名...

  • 使用os pipe管道使python fork多进程之间通信

    管道(pipe) 管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功...

  • 线程-进程间通信(操作系统、java、android)最全总结!

    大纲 操作系统进程间通信 进程的通信机制主要有:管道、有名管道、消息队列、信号量、共享空间、信号、套接字。 操作系...

网友评论

      本文标题:PHP进程间通信--有名管道

      本文链接:https://www.haomeiwen.com/subject/tbudmxtx.html