PHP Swoole 基本使用

2020-07-16 10:05:52
背景

專案中使用的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 closen";
    }
 
    public function onError()
    {
        echo "Connect failedn";
    }
 
}

注意問題

'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 基本使用的詳細內容,更多請關注TW511.COM其它相關文章!