一、为什么你要用swoole,能解决你项目中的哪些痛点?二、你是如何通过swoole提升性能的,怎么做的?三、swoole里的协程是什么,怎么用?为什么协程可以提高并发?四、用了swoole以后,会不会发生内存泄漏?如果发生了怎么解决?五、swoole和php-fpm的区别
一、为什么你要用swoole,能解决你项目中的哪些痛点?swoole是一个网络通讯和异步IO的引擎,一个基础库;
swoole相比于apache/fpm,主要节省了PHP框架和全局对象每次创建销毁带来的性能开销,是进程常驻内存型。
二、你是如何通过swoole提升性能的,怎么做的?进程常驻内存:
swoole本身是进程常驻内存,在进程启动的时候就将PHP框架等代码读取并编译完成,不需要每次启动的时候都执行编译步骤,大大降低了脚本的运行时间;
连接池
php-fpm的模式php因为每次请求结束时都会销毁所有资源,因此无法使用连接池;而基于swoole的进程常驻内存模式,可以通过连接池的方式来加速程序,
使用连接池既可以降低程序的响应时间,又可以有效保护后端资源。
可以使用协程处理异步IO
当开发中需要去请求多处的数据,而每一块的数据单独请求都要花较长时间,常规的php-fpm是阻塞式运行,无法对这类型的数据处理进行加速;而基于swoole的程序,可以将这类的业务并行化处理,并行去请求后端的数据源,能够大大优化了此类业务的运行时间。
协程是通过协作而不是抢占的方式来进行切换,它创建和切换对内存等资源比线程小的多(可以理解为更小的线程);
协程的使用是通过SwooleCoroutine或者Co命名空间短命名简化类名来创建;
协程可以异步处理任务,支持并发,并且资源消耗小。
四、用了swoole以后,会不会发生内存泄漏?如果发生了怎么解决?swoole由于是常驻内存,一旦资源加载进入后,会一直存在于内存中。对于局部变量,swoole会在回调函数结束后自动释放;对于全局变量(lobal声明的变量,
static声明的对象属性或者函数内的静态变量和超全局变量),swoole不会自动释放;因此操作不好会发生内存泄漏。
解决:
在onClose回调内清理变量;
swoole提供了max_request和max_task_request机制:进程完成指定数量的任务后,会自动退出,达到释放资源和内存的目的;而后manager进程会重新拉起新worker/task进程来继续处理任务。
使用限制: max_request只能用于同步阻塞、无状态的请求响应式服务器程序; 纯异步的Server不应当设置max_request 使用base模式时max_request是无效的五、swoole和php-fpm的区别
PHP-FPM:
早期版本的 PHP 并没有内置的 WEB 服务器,而是提供了 SAPI(Server API)给第三方做对接。现在非常流行的 php-fpm 就是通过 FastCGI 协议来处理 PHP 与第三方 WEB 服务器之间的通信。比如 Nginx + php-fpm 的组合,这种方式运行的 fpm 是 Master/Worker 模式,启动一个 Master 进程监听来自 Nginx 的请求,再 fork 多个 Worker 进程处理请求。每个 Worker 进程只能处理一个请求,单一进程的生命周期大体如下:
初始化模块。
初始化请求。此处请求是请求 PHP 执行代码的意思,并非 HTTP 的请求。
执行 PHP 脚本。
结束请求。
关闭模块。
多进程模型是依赖进程数来解决并发问题,一个进程只能处理一个连接,当启动大量进程,进程调度消耗可能占 CPU 的百分之几十甚至 100%,比如 C10K 问题,多进程模型就力不从心了。
Swoole:
Swoole 采用的也是 Master/Worker 模式,不同的是 Master 进程有多个 Reactor 线程,Master 只是一个事件发生器,负责监听 Socket 句柄的事件变化。Worker 以多进程的方式运行,接收来自 Reactor 线程的请求,并执行回调函数(PHP 编写的)。启动 Master 进程的流程大致是:
初始化模块。
初始化请求。因为 swoole 需要通过 cli 的方式运行,所以初始化请求时,不会初始化 PHP 的全局变量,如 $_SERVER, $_POST, $_GET 等。
执行 PHP 脚本。包括词法、语法分析,变量、函数、类的初始化等,Master 进入监听状态,并不会结束进程。
Swoole 加速的原理
由 Reactor(epoll 的 IO 复用方式)负责监听 Socket 句柄的事件变化,解决高并发问题。
通过内存常驻的方式节省 PHP 代码初始化的时间,在使用笨重的框架时,用 swoole 加速效果是非常明显的。
对比不同
PHP-FPM
Master 主进程 / Worker 多进程模式。
启动 Master,通过 FastCGI 协议监听来自 Nginx 传输的请求。
每个 Worker 进程只对应一个连接,用于执行完整的 PHP 代码。
PHP 代码执行完毕,占用的内存会全部销毁,下一次请求需要重新再进行初始化等各种繁琐的操作。
只用于 HTTP Server。
Swoole
Master 主进程(由多个 Reactor 线程组成)/ Worker 多进程(或多线程)模式
启动 Master,初始化 PHP 代码,由 Reactor 监听 Socket 句柄的事件变化。
Reactor 主线程负责子多线程的均衡问题,Manager 进程管理 Worker 多进程,包括 TaskWorker 的进程。
每个 Worker 接受来自 Reactor 的请求,只需要执行回调函数部分的 PHP 代码。
只在 Master 启动时执行一遍 PHP 初始化代码,Master 进入监听状态,并不会结束进程。
不仅可以用于 HTTP Server,还可以建立 TCP 连接、WebSocket 连接。



