zoukankan      html  css  js  c++  java
  • PHP中多线程处理

    php多进程处理

    往往我们会碰到一个情况,需要写一个脚本,这个脚本要处理的数据量极大,单进程处理脚本非常慢,那么这个时候就会想到使用多进程或者多线程的方式了。

    我习惯使用多进程的方式,php中使用多进程的时候需要使用pcntl,pcntl的使用可以看这个PHP的pcntl多进程

    但是这里有一个问题,一个主进程把任务分成n个部分,然后把任务分配给多个子进程,但是任务可能是有返回值的,所有的子进程处理完返回值以后需要把返回值返回给主进程。

    这个就涉及到了进程间通信了。进程间通信可以使用的方法当然很多了,比如用redis,用数据库,用文件等。

    php中最简单的要算shmop相关函数了。

    • shmop_open
    • shmop_read
    • shmop_write
    • shmop_size
    • shmop_delete

    那怎么让一个类很容易有多进程处理的能力呢?可以使用php的trait,创建一个PcntlTrait,所有需要有多进程处理功能的类就use 这个trait就行。

    PcntlTrait代码如下:

    <?php namespace App\Console\Commands;
    
    trait PcntlTrait
    {
        private $workers = 1;
    
        public function worker($count)
        {
            $this->workers = $count;
        }
    
        public function pcntl_call($all, \Closure $callback)
        {
            $perNum = ceil(count($all) / $this->workers);
    
            $pids = [];
            for($i = 0; $i < $this->workers; $i++){
                $pids[$i] = pcntl_fork();
                switch ($pids[$i]) {
                    case -1:
                        echo "fork error : {$i} \r\n";
                        exit;
                    case 0:
                        $data = [];
                        try {
                            $data = $callback(array_slice($all, $i * $perNum, $perNum));
                        } catch(\Exception $e) {
                            echo ($e->getMessage());
                        }
    
                        $shm_key = ftok(__FILE__, 't') . getmypid();
                        $data = json_encode($data);
                        $shm_id = shmop_open($shm_key, "c", 0777, strlen($data) + 10);
                        shmop_write($shm_id, $data, 0);
                        shmop_close($shm_id);
                        exit;
                    default:
                        break;
                }
            }
    
            // only master process will go into here
            $ret = [];
            foreach ($pids as $i => $pid) {
                if($pid) {
                    pcntl_waitpid($pid, $status);
    
                    $shm_key = ftok(__FILE__, 't') . $pid;
                    $shm_id = shmop_open($shm_key, "w", 0, 0);
    
                    $data = trim(shmop_read($shm_id, 0, shmop_size($shm_id)));
                    $data = json_decode($data, true);
                    $ret = array_merge($ret, $data);
                    @shmop_close($shm_id);
                    @shmop_delete($shm_id);
                }
            }
    
            return $ret;
        }
    }

    它有两个参数,第一个参数为传入数组,第二个参数为数组处理函数。

    它的具体使用通过下面这个测试用例可以看出:

    <?php
    
    use App\Console\Commands\PcntlTrait;
    
    class PcntlImp
    {
            use PcntlTrait;
    }
    
    class TestPcntlTrait extends \TestCase
    {
        public function setup()
        {
            $this->createApplication();
        }
    
        public function testPcntlCall()
        {
            $arr = [1,2,3,4,5,6,7,8,9,10];
    
            $imp = new \PcntlImp();
            $imp->worker(2);
    
            $data = $imp->pcntl_call($arr, function($info){
                if (empty($info)){
                    return [];
                }
    
                $ret = [];
                foreach ($info as $item) {
                    $ret[] = $item * $item;
                }
                return $ret;
            });
    
            $this->assertEquals(10, count($data));
            $this->assertEquals(25, $data[4]);
        }
    }
  • 相关阅读:
    BZOJ2435 NOI2011道路修建
    BZOJ2431 HAOI2009逆序对数列(动态规划)
    BZOJ2456 mode
    BZOJ2324 ZJOI2011营救皮卡丘(floyd+上下界费用流)
    BZOJ2303 APIO2011方格染色(并查集)
    BZOJ2299 HAOI2011向量(数论)
    BZOJ2169 连边(动态规划)
    BZOJ2159 Crash的文明世界(树形dp+斯特林数)
    洛谷 P1306 斐波那契公约数 解题报告
    洛谷 P2389 电脑班的裁员 解题报告
  • 原文地址:https://www.cnblogs.com/lq527/p/6197381.html
Copyright © 2011-2022 走看看