[toc] >[info] 用于替代PHP-FPM同步阻塞的模式。实现异步非阻塞高性能的web服务 ## 优势 * 性能的数十倍的提升; * 可以在`Apache`/`Nginx`等传统WEB服务器和`Swoole`之间切换部署; * **在此模式下,还可以直接在控制器中调用`swoole/server`的方法,比如定时,任务异步投递,协程等等** > 本例使用的是thinkphp5.1的扩展包, 本文只做实操记录与源码分析,详细内容查看 [ThinkPHP 5.1 Swoole 快速上手指南](https://ihavenolimitations.xyz/thinkphp/think-swoole/722895)。 ~~~ composer require topthink/think-swoole # 启动 php think swoole ~~~ ## 理解性能提升 原生的php-fpm是同步阻塞模式的,PHP-FPM 会创建一个主进程,控制何时以及如何将HTTP请求转发给一个或多个子进程处理。PHP-FPM主进程还控制着什么时候创建(处理Web应用更多的流量)和销毁(子进程运行时间太久或不再需要了)。 <br /> 而swoole-httpserver是异步非阻塞模式的。它常驻于内存,并基于事件回调。 **我们先通过ab压测分别压测不同模式下的同一PHP程序** `这里压的是开了debug的thinkphp5.1默认页面` ~~~ // -n 2000表示总请求数为2000 // -c 100 表示并发用户数为100 // -k 使用HTTP的KeepAlive特性 ab -c 100 -n 2000 -k http://127.0.0.1:9501/ ~~~ 关于ab压测工具的安装与使用:[https://cloud.tencent.com/developer/article/1333772](https://cloud.tencent.com/developer/article/1333772) **在php-fpm模式下** ![](https://i.loli.net/2019/04/12/5cb055468f405.png) **在swoole-httpserver模式下** ![](https://i.loli.net/2019/04/12/5cb056319e8c7.png) >[info] 如图所见,几十倍近百倍的性能提升。**注意:**这可不是压 `hello world` 哦,而是压加载了93个文件的框架默认页。 >[danger] 注意,异步或协程模式下的httpserver中不能有同步阻塞的代码,否则就会变成同步,如果变成同步以后,性能还远远不如**PHP-fpm** > [PHP中哪些函数是同步阻塞的](https://wiki.swoole.com/wiki/page/474.html)。 **目前TP-swoole2.0版本对底层数据库的协程是还不支持的!到3.0版本才支持** ### 异步的优势 * 高并发,同步阻塞IO模型的并发能力依赖于进程/线程数量,例如`php-fpm`开启了200个进程,理论上最大支持的并发能力为200。如果每个请求平均需要100ms,那么应用程序就可以提供2000qps。异步非阻塞的并发能力几乎是无限的,可以发起或维持大量并发TCP连接 * 无IO等待,同步模型无法解决`IOWait`很高的场景,如上述例子每个请求平均要10s,那么应用程序就只能提供20qps了。而异步程序不存在IO等待,所以无论请求要花费多长时间,对整个程序的处理能力没有任何影响 ### 同步的优势 * 编码简单,同步模式编写/调试程序更轻松 * 可控性好,同步模式的程序具有良好的过载保护机制,如在下面的情况异步程序就会出问题 * Accept保护,同步模式下一个TCP服务器最大能接受`进程数+Backlog`个TCP连接。一旦超过此数量,Server将无法再接受连接,客户端会连接失败。避免服务器Accept太多连接,导致请求堆积 ## 配置问题 如果应用目录`application`因为使命名空间更具语义化而更改为`app`的话,要手动指定配置项`file_monitor_path` ~~~ 'file_monitor_path' => ['/home/wwwroot/XGservice/app/','/home/wwwroot/XGservice/config/'], // 文件监控目录 默认监控application和config目录 ~~~ >[warning] 切换成swoole模式后,应用就不是从入口文件index.php启动了。而是在onWorkerStart回调中启动,启动的是think/swoole/application(继承自think/app) ## 原生实现 [Http\\Server](https://wiki.swoole.com/wiki/page/327.html) ~~~ use Swoole\Http\Server; $http = new Server("0.0.0.0", 9501); $http->on('request', function ($request, $response) { $response->end("<h1>Hello Swoole. #".rand(1000, 9999)."</h1>"); }); $http->start(); ~~~ `Http\Server`继承自`Server`,是一个的`Http`服务器实现。`Http\Server`支持同步和异步2种模式。无论是同步模式还是异步模式,`Http\Server`都可以维持大量`TCP`客户端连接。同步/异步仅仅体现在对请求的处理方式上。 >[info] `Http/WebSocket`服务器都是继承自`Server`,所以`Server`提供的`API`,如`task/finish/tick`等都可以使用 ## 扩展源码分析 `/vendor/topthink/think-swoole/src/command/Swoole.php` 1. 服务启动前,有钩子监听后续可增加钩子扩展业务 ~~~ $hook = Container::get('hook'); $hook->listen("swoole_server_start", $swoole); ~~~ 2. **在Worker进程启动时也预留了钩子,同时调用了config里面配置的闭包或者类文件(run方法)** ~~~ // vendor/topthink/think-swoole/src/Http.php:142 // 此步就是原生中on回调onWokerstart //onWokerstart预留的钩子 $hook = Container::get('hook'); $hook->listen('swoole_worker_start', ['server' => $server, 'worker_id' => $worker_id]); // 闭包或者类文件 $wokerStart = Config::get('swoole.wokerstart'); ~~~ >[info] swoole是基于事件的。thinkphp-swoole在显式的指定了onWorkerStart(server的回调事件),onRequest(httpServer的回调事件),其它事件可以在 3. 通过 [Process::kill](https://wiki.swoole.com/wiki/page/p-process.html) 实现 `stop`,`reload` 管理 ~~~ # 检测进程是否存在,不会发送信号 Process::kill($pid, 0); # stop Process::kill($pid, SIGTERM); # reload 柔性重启 Process::kill($pid, SIGUSR1); # start /* $swoole = new Swoole\Http\Server("0.0.0.0", 9501,SWOOLE_PROCESS, SWOOLE_SOCK_TCP); */ $swoole->option($this->config); $swoole->start(); # restart # 就是stop之后再start ~~~ 4. 响应事件 这段代码写得可以,值得借鉴 ``` // vendor/topthink/think-swoole/src/Server.php:93 protected $event = ['Start', 'Shutdown', 'WorkerStart', 'WorkerStop', 'WorkerExit', 'Connect', 'Receive', 'Packet', 'Close', 'BufferFull', 'BufferEmpty', 'Task', 'Finish', 'PipeMessage', 'WorkerError', 'ManagerStart', 'ManagerStop', 'Open', 'Message', 'HandShake', 'Request']; // 循环判断,如果有当前回调方法就注册 foreach ($this->event as $event) { if (method_exists($this, 'on' . $event)) { $this->swoole->on($event, [$this, 'on' . $event]); } } ``` ## `Cookie`和`Session` >[danger] 由于`Swoole`的特殊性,扩展本身接管了`Cookie`类和`Session`类的处理,因此不要调用(包括依赖注入)`think\Cookie`和`think\Session`类,而应该改为`think\swoole\Cookie`类和`think\swoole\Session`类。并且所有PHP内置的`session`函数都是无效的,`think-swoole`扩展单独封装了一个`Session`类,和系统的`Session`机制无关。 扩展虽然提供了基于`swoole_table`和`ThinkPHP`缓存的混合解决方案。但建议使用缓存机制来处理`Session`,如果你的应用比较大,则应该配置使用`redis`之类的缓存机制更加适合,直接在你的`cache.php`中定义相关缓存配置即可。 >[info] 其实在实际的项目开发中,更多的是开发api接口,而且都是基于JWT的身份验证,cookie和session反而用得很少。