背景
项目中使用的PHP,但由于长耗时的任务,前端提交以后,需要服务端异步响应。 服务器异步有多种方案,包括MQ,fsocket,Swoole等。 Swoole 使用纯 C 语言编写,提供了 PHP 语言的异步多线程服务器,异步 TCP/UDP 网络客户端,异步 MySQL,异步 Redis,数据库连接池,AsyncTask,消息队列,毫秒定时器,异步文件读写,异步DNS查询。Swoole内置了Http/WebSocket服务器端/客户端、Http2.0服务器端。 最重要的是,完美支持PHP语言。于是使用Swoole搭建了一个异步服务器,提供异步响应,推送,定时任务等一系列工作。
安装
Swoole是C语言编写,采用编译安装的方式。
安装依赖项有:
代码语言:javascript复制php-5.3.10 或更高版本gcc-4.4 或更高版本
代码语言:javascript复制makeautoconfpcre (centos系统可以执行命令:yum install pcre-devel)
安装方式:
代码语言:javascript复制phpize #如果命令不存在 请在前面加上php的实际路径./configuremake sudo make install
编译完成以后,需要在php.ini中添加扩展
代码语言:javascript复制extension=swoole.so
使用
服务端
代码语言:javascript复制class Server{ private $serv; public function __construct() { $this->serv = new swoole_server("0.0.0.0", 9501); $this->serv->set(array( 'daemonize' => 1, 'max_request' => 10000, 'task_worker_num' => 1, "task_ipc_mode " => 3 , 'open_length_check' => true, 'dispatch_mode' => 1, 'package_length_type' => 'N', 'package_length_offset' => 0, '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) { }
客户端
代码语言:javascript复制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, '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 . "
"; } public function onConnect($cli) { $data = pack('N', strlen($data)) . $data; $cli->send($data); $cli->close(); } public function onClose( $cli) { echo "Connection close
"; } public function onError() { echo "Connect failed
"; }}
注意问题
代码语言:javascript复制'open_length_check' => true,'package_length_type' => 'N','package_length_offset' => 0, 'package_body_offset' => 4, 'package_max_length' => 2000000,
这几个是定义帧定界的,因为Swoole的客户端和服务器端通信是TCP连接的,因此得给帧定界符,有多种帧定界方式,具体参考Swoole官方文档。这里其中是用头额外加长度的方式。