背景
專案中使用的PHP,但由於長耗時的任務,前端提交以後,需要服務端非同步回應。
伺服器非同步有多種方案,包括MQ,fsocket,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 onError() { echo "Connect failed\n"; } }
注意問題
'open_length_check' => true, 'package_length_type' => 'N', 'package_length_offset' => 0, //第N个字节是包长度的值 'package_body_offset' => 4, //第几个字节开始计算长度 'package_max_length' => 2000000, //协长度
這幾個是定義幀定界的,因為Swoole的客戶端和伺服器端通訊是TCP連接的,因此得給幀定界符,有多種幀定界方式,具體參考Swoole官方文檔。這裡其中是用頭額外加長度的方式。
以上是PHP Swoole 基本使用的詳細內容。更多資訊請關注PHP中文網其他相關文章!