PHP使用Swoole搭建一个异步服务

项目中使用的PHP,但由于长耗时的任务,前端提交以后,需要服务端异步响应。

服务器异步有多种方案,包括MQ,fsocket,Swoole等。 (推荐学习: swoole视频教程)

Swoole 使用纯 C 语言编写,提供了 PHP 语言的异步多线程服务器,异步 TCP/UDP 网络客户端,异步 MySQL,异步 Redis,数据库连接池,AsyncTask,消息队列,毫秒定时器,异步文件读写,异步DNS查询。 Swoole内置了Http/WebSocket服务器端/客户端、Http2.0服务器端。

最重要的是,完美支持PHP语言。于是使用Swoole搭建了一个异步服务器,提供异步响应,推送,定时任务等一系列工作。

Swoole是C语言编写,采用编译安装的方式。

安装依赖项有:

php-5.3.10 或更高版本

gcc-4.4 或更高版本

make

autoconf

pcre (centos系统可以执行命令:yum install pcre-devel)

安装方式:

phpize #如果命令不存在 请在前面加上php的实际路径

./configure

make 

sudo make install

编译完成以后,需要在php.ini中添加扩展

extension=swoole.so

服务端

class Server{

    private $serv;

    public function __construct() {

        $this->serv = new swoole_server("0.0.0.0", 9501);

        $this->serv->set(array(

            //'worker_num' => 1,  //一般设置为服务器CPU数的1-4倍

            'daemonize' => 1,  //以守护进程执行

            'max_request' => 10000,

            'task_worker_num' => 1,  //task进程的数量

            "task_ipc_mode " => 3 ,  //使用消息队列通信,并设置为争抢模式

            'open_length_check'    => true,

            'dispatch_mode'        => 1,

            'package_length_type'  => 'N',  //这个很关键,定位包头的

            'package_length_offset' => 0,      //第N个字节是包长度的值

            'package_body_offset'  => 4,      //第几个字节开始计算长度

            'package_max_length'    => 2000000,  //协议最大长度

            "log_file" => "/tmp/swoole_test.log"  //日志

        ));

        $this->serv->on('Receive', array($this, 'onReceive'));

        $this->serv->on('Task', array($this, 'onTask'));

        $this->serv->on('Finish', array($this, 'onFinish'));

        $this->serv->start();

    }

    public function onReceive( swoole_server $serv, $fd, $from_id, $data ) {

        //放入任务队列,开始执行

        $task_id = $serv->task( $data );

    }

    public function onTask($serv,$task_id,$from_id, $data) {

      //做一些事情

    }

客户端

class Client{

    private $client, $ip, $port, $params;

    public function __construct($ip, $port, $params)

    {

        $this->ip = $ip;

        $this->port = $port;

        $this->params = $params;

        $this->client = new swoole_client(SWOOLE_SOCK_TCP, SWOOLE_SOCK_ASYNC);

        $this->client->set(array(

            'open_length_check'    => true,

            'package_length_type'  => 'N',

            'package_length_offset' => 0,      //第N个字节是包长度的值

            'package_body_offset'  => 4,      //第几个字节开始计算长度

            'package_max_length'    => 2000000,  //协议最大长度

        ));

        //设置事件回调函数

        $this->client->on('Connect', array($this, 'onConnect'));

        $this->client->on('Receive', array($this, 'onReceive'));

        $this->client->on('Close', array($this, 'onClose'));

        $this->client->on('Error', array($this, 'onError'));

        //发起网络连接

        $this->client->connect($ip, $port, 3);

    }

    public function onReceive( $cli, $data ) {

        echo "Received: " . $data . "\n";

    }

    public function onConnect($cli) {

        $data = pack('N', strlen($data)) . $data;

        $cli->send($data);

        $cli->close();

    }

    public function onClose( $cli)

    {

        echo "Connection close\n";

    }

    public function one rror()

    {

        echo "Connect failed\n";

    }

}

 

上一篇:contos7 yum安装php7.2与swoole


下一篇:关于php命令行模式介绍