目录

【了解Swoole】

【PHP中使用Swoole案例演示】

安装Swoole扩展

Swoole实现TCP请求

Swoole实现UDP请求

Swoole实现HTTP请求

Swoole实现WebSocket聊天室功能

Swoole执行异步任务 (Task)

Swoole实现Redis服务器

PHPStorm中添加swoole智能提示


【了解Swoole】

为什么要学习使用swoole,首先说说PHP存在的缺陷:

  • 不能常驻内存
  • 对多线程支持不好,不支持协程
  • 解释性语言

Swoole官网: Swoole - PHP 协程框架 是这么说明的:

Swoole 使 PHP 开发人员可以编写高性能高并发的 TCP、UDP、Unix Socket、HTTP、 WebSocket 等服务,让 PHP 不再局限于 Web 领域。Swoole4 协程的成熟将 PHP 带入了前所未有的时期, 为性能的提升提供了独一无二的可能性。Swoole 可以广泛应用于互联网、移动通信、云计算、 网络游戏、物联网(IOT)、车联网、智能家居等领域。使用 PHP + Swoole 可以使企业 IT 研发团队的效率大大提升,更加专注于开发创新产品。

Swoole的特性:

  • Swoole使用C/C++语言编写,提供了PHP语言的异步多线程服务器、异步TCP/UDP网络客户端、异步MySQL、异步Redis、数据库连接池、AsyncTask、 消息队列、毫秒定时器、异步文件读写、异步DNS查询。Swoole内置了Http/WebSocket服务器端/客户端、Http2.0 服务器端。
  • 除了异步IO的支持之外, Swoole为PHP多进程的模式设计了多个并发数据结构和IPC通信机制,可以大大简化多进程并发编程的工作。其中包括了原子计数器、Table、 Channel、Lock、进程间通信IPC等丰富的功能特性。
  • Swoole4.0支持了类似Go语言的协程,可以使用完全同步的代码实现异步程序。PHP 代码无需额外增加任何关键词,底层自动进行协程调度,实现异步IO。

Swoole的优点:

  • 高性能的异步
  • 提供了网络通信的能力
  • 方便地开发 Http、WebSocket、TCP、UDP 等应用
  • 协程

【PHP中使用Swoole案例演示】

安装Swoole扩展

下载swoole,进入swoole目录,编译安装:

phpize
./configure
make && make install
然后给php.ini加入swoole.so

查看是否编译成功:php -m, 或者 php --ri swoole

查看phpinfo()效果如下:

Swoole源码包里面 examples/server 目录下有个 eho.php,运行:php echo.php
使用 netstat -anp | grep 9501 查看端口情况(Mac不能带参数p,或者用 lsof -i:9501)

Swoole实现TCP请求

官网资料:https://wiki.swoole.com/#/start/start_tcp_server

<?php

class TCP {
    private $server = null;

    public function __construct() {
        $this->server = new Swoole\Server("127.0.0.1", 9501);
        $this->server->set([
            'worker_num' => 4,
            'max_request' => 50,
        ]);
        $this->server->on('Connect', [$this, "onConnect"]);
        $this->server->on('Receive', [$this, "onReceive"]);
        $this->server->on('Close', [$this, "onClose"]);

        //启动服务器
        $this->server->start();
    }

    public function onConnect($server, $fd) {
        echo "客户端id: {$fd} 链接.\n";
    }

    public function onReceive($server, $fd, $from_id, $data) {
        $server->send($fd, "发送的数据:" . $data);
    }

    public function onClose($server, $fd) {
        echo "客户端id: {$fd}关闭.\n";
    }

}

new TCP();

  • 服务器可以同时被成千上万个客户端连接,$fd 就是客户端连接的唯一标识符。
  • 调用 $server->send() 方法向客户端连接发送数据,参数就是 $fd 客户端标识符。
  • 调用 $server->close() 方法可以强制关闭某个客户端连接。
  • 客户端可能会主动断开连接,此时会触发 onClose 事件回调。 

Swoole实现UDP请求

官网资料:https://wiki.swoole.com/#/start/start_udp_server

<?php

class UDP {
    private $server = null;

    public function __construct() {
        $this->server = new Swoole\Server('127.0.0.1', 9502, SWOOLE_PROCESS, SWOOLE_SOCK_UDP);
        $this->server->set([
            'worker_num' => 4,
            'max_request' => 50,
        ]);
        $this->server->on('Packet', [$this, "onPacket"]);

        //启动服务器
        $this->server->start();
    }

    public function onPacket($server, $data, $clientInfo) {
        var_dump($clientInfo);
        $server->sendto($clientInfo['address'], $clientInfo['port'], "Server:{$data}");
    }
}

new UDP();

UDP 服务器与 TCP 服务器不同,UDP 没有连接的概念。启动 Server 后,客户端无需 Connect,直接可以向 Server 监听的 9502 端口发送数据包。对应的事件为 onPacket。

  • $clientInfo 是客户端的相关信息,是一个数组,有客户端的 IP 和端口等内容。
  • 调用 $server->sendto 方法向客户端发送数据。

如何理解TCP和UDP:TCP 是面向连接的、可靠的、只支持点对点通信;UDP 是无连接的、不可靠的、支持一对一、一对多、多对一、多对多的通信模式。

TCP就像是两个人打电话,你必须听清楚对方讲的什么才能知道回复什么;而UDP就像是马路上的广播,它不会在乎你有没有听到,错过就是错过了。

Swoole实现HTTP请求

官网资料:https://wiki.swoole.com/#/start/start_http_server

<?php

class HTTP {
    private $http = null;

    public function __construct() {
        $this->http = new Swoole\Http\Server('0.0.0.0', 9503);
        $this->http->set([
            'enable_static_handler' => true,
            'document_root' => "./static",
        ]);
        $this->http->on('Request', [$this, "onRequest"]);

        //启动服务器
        $this->http->start();
    }

    public function onRequest($request, $response) {
        var_dump($request->get, $request->post);


        $response->header('Content-Type', 'text/html; charset=utf-8');
        $response->end('Hello Swoole.' . json_encode($request->get));
    }
}

new HTTP();

通过浏览器访问http根目录,并且指定参数:

通过浏览器直接访问静态资源html:

HTTP 服务器只需要关注请求响应即可,所以只需要监听一个 onRequest 事件。当有新的 HTTP 请求进入就会触发此事件。事件回调函数有 2 个参数,一个是 $request 对象,包含了请求的相关信息,如 GET/POST 请求的数据。另外一个是 response 对象,对 request 的响应可以通过操作 response 对象来完成。$response->end() 方法表示输出一段 HTML 内容,并结束此请求。

  • 0.0.0.0 表示监听所有 IP 地址,一台服务器可能同时有多个 IP,如 127.0.0.1 本地回环 IP、192.168.1.100 局域网 IP、210.127.20.2 外网 IP,这里也可以单独指定监听一个 IP;
  • 9501 监听的端口,如果被占用程序会抛出致命错误,中断执行。

Swoole实现WebSocket聊天室功能

官网资料:https://wiki.swoole.com/#/start/start_ws_server

<?php

class WS {
    private $http = null;

    public function __construct() {
        //第三和第四个参数,是在https的情况下需要配置的
        //$this->http = new Swoole\WebSocket\Server('0.0.0.0', 9502, SWOOLE_PROCESS, SWOOLE_SOCK_TCP | SWOOLE_SSL);
        $this->http = new Swoole\WebSocket\Server('0.0.0.0', 9502);

        $this->http->set([
            //下面的部分也是用来配置https的ssl证书的
            'ssl_cert_file' => "",
            'ssl_key_file' => "",

            'enable_static_handler' => true,
            'document_root' => "./static",
        ]);
        $this->http->on('Open', [$this, "onOpen"]);
        $this->http->on('Message', [$this, "onMessage"]);
        $this->http->on('Close', [$this, "onClose"]);

        //启动服务器
        $this->http->start();
    }

    public function onOpen($ws, $request) {
        $ws->push($request->fd, "hello,welcome\n");
    }

    public function onMessage($ws, $frame) {
        echo "Message: {$frame->data}\n";
        foreach ($ws->connections as $fd) {
            if ($fd == $frame->fd) {
                $ws->push($fd, "我: {$frame -> data}");
            } else {
                $ws->push($fd, "对方:{$frame -> data}");
            }
        }
    }

    public function onClose($ws, $fd) {
        echo "client:{$fd} is closed\n";
    }
}

new WS();

客户端JS代码:

var wsServer = 'ws://127.0.0.1:9502';
var websocket = new WebSocket(wsServer);
websocket.onopen = function (res) {
    $("#welcome").append(
        "<h1>连接成功!欢迎</h1>"
    );
};
websocket.onclose = function (res) {
    $("#message").append(
        "<h3>" + res.data + "</h3>"
    );
};
websocket.onmessage = function (res) {
    $("#message").append(
        "<h3>" + res.data + "</h3>"
    );
};
websocket.onerror = function (res, e) {
    $("#message").append(
        "<h3>" + res + "</h3>"
    );
};
function send() {
    websocket.send($("#input").val());
}

如何理解WebSocket:是一种基于 TCP 的轻量级网络通信协议,在地位上是与 HTTP“平级”的。HTTP 难以应用在动态页面、即时消息、网络游戏等要求“实时通信”的领域;WebSocket 客户端和服务器都可以随时向对方发送数据。

  • 客户端向服务器端发送信息时,服务器端触发 onMessage 事件回调。
  • 服务器端可以调用 $server->push() 向某个客户端(使用 $fd 标识符)发送消息。

Swoole执行异步任务 (Task)

官网资料:https://wiki.swoole.com/#/start/start_task

<?php

$http = new Swoole\Http\Server('0.0.0.0', 9501);

$http->set([
//    'worker_num' => 1,
    //如果要使用 Task ,需要先设置 task_worker_num ,它代表的是开启的 Task 进程数量。
    'task_worker_num' => 4,
]);

$http->on('Request', function ($request, $response) use ($http) {
    echo "接收到了请求", PHP_EOL;
    $response->header('Content-Type', 'text/html; charset=utf-8');

    $http->task("发送邮件");
    $http->task("发送广播");
    $http->task("执行队列");

    $response->end('<h1>Hello Swoole. #' . rand(1000, 9999) . '</h1>');
});

//处理异步任务(此回调函数在task进程中执行)
//Task 事件是用于处理任务的,可以根据传递过来的 $data 内容进行处理。
$http->on('Task', function ($serv, $task_id, $reactor_id, $data) {
    $sec = rand(1, 5);
    echo "New AsyncTask[id={$task_id}] sleep sec: {$sec}" . PHP_EOL;
    sleep($sec);
    //返回任务执行的结果
    $serv->finish("{$data} -> OK");
});

//处理异步任务的结果(此回调函数在worker进程中执行)
//Finish 事件是监听任务结束,当执行的任务结束后,就会调用这个事件回调,可以进行后续的处理。如果你的任务没有后续的处理,那么我们也可以不去监听这个事件。
$http->on('Finish', function ($serv, $task_id, $data) {
    echo "AsyncTask[{$task_id}] Finish: {$data}" . PHP_EOL;
});

echo "服务启动", PHP_EOL;
$http->start();

Swoole实现Redis服务器

官网资料:https://wiki.swoole.com/#/redis_server

//使用 setHandler() 方法来监听 Reids 命令,在这里我们看到了熟悉的 get、set 等命令的定义。
$server->setHandler('GET', function ($fd, $data) use ($server) {
    if (count($data) == 0) {
        return $server->send($fd, Server::format(Server::ERROR, "ERR wrong number of arguments for 'GET' command"));
    }

    $key = $data[0];
    //指定了 $server->data ,可以将它看成是一个数据源,直接使用的就是一个文件,
    //直接在当前测试环境目录下创建一个叫做 db 的空文件就可以了。
    if (empty($server->data[$key])) {
        //使用 send() 方法来返回响应的命令信息,并通过 format() 方法格式化返回的响应数据。
        return $server->send($fd, Server::format(Server::NIL));
    } else {
        return $server->send($fd, Server::format(Server::STRING, $server->data[$key]));
    }
});

以上内容完整代码参考: https://gitee.com/rxbook/thinkphp-demo-2023/tree/master/swoole_test1

PHPStorm中添加swoole智能提示

备注,默认情况下PHPStorm中是不提示Swoole相关函数信息的,比如下面这样:

如果要给PHPStorm中添加swoole智能提示,方法如下:

下载函数库 git clone https://github.com/eaglewu/swoole-ide-helper.git

加载方式1: 右键External Libraries,选择Configure PHP Include Path, 选择下载好的swoole-ide-helper目录,点击确定, 只提供给本项目使用。

加载方式2: 将代码包含到PhpStorm的Settings->Languages & Frameworks->PHP->Include path里面, 提供给本机使用。

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐