概念
Hyperf 是运行于 Swoole 5
的协程和 Swow
协程之上的,这也是 Hyperf 能提供高性能的其中一个很大的因素。
PHP-FPM 的运作模式
在聊协程是什么之前,我们先聊聊传统 PHP-FPM
架构的运作模式,PHP-FPM
是一个多进程的 FastCGI
管理程序,是绝大多数 PHP
应用所使用的运行模式。假设我们使用 Nginx
提供 HTTP
服务(Apache
同理),所有客户端发起的请求最先抵达的都是 Nginx
,然后 Nginx
通过 FastCGI
协议将请求转发给 PHP-FPM
处理,PHP-FPM
的 Worker 进程
会抢占式的获得 CGI 请求进行处理,这个处理指的就是,等待 PHP
脚本的解析,等待业务处理的结果返回,完成后回收子进程,这整个的过程是阻塞等待的,也就意味着 PHP-FPM
的进程数有多少能处理的请求也就是多少,假设 PHP-FPM
有 200
个 Worker 进程
,一个请求将耗费 1
秒的时间,那么简单的来说整个服务器理论上最多可以处理的请求也就是 200
个,QPS
即为 200/s
,在高并发的场景下,这样的性能往往是不够的,尽管可以利用 Nginx
作为负载均衡配合多台 PHP-FPM
服务器来提供服务,但由于 PHP-FPM
的阻塞等待的工作模型,一个请求会占用至少一个 MySQL
连接,多节点高并发下会产生大量的 MySQL
连接,而 MySQL
的最大连接数默认值为 100
,尽管可以修改,但显而易见该模式没法很好的应对高并发的场景。
异步非阻塞系统
在高并发的场景下,异步非阻塞就显得优势明显了,直观的优点表现就是 Worker 进程
不再同步阻塞的去处理一个请求,而是可以同时处理多个请求,无需 I/O
等待,并发能力极强,可以同时发起或维护大量的请求。那么最直观的缺点大家可能也都知道,就是永无止境的回调,业务逻辑必须在对应的回调函数内实现,如果业务逻辑存在多次的 I/O
请求,则会存在很多层的回调函数,下面示例一段 Swoole 1.x
下的异步回调型的伪代码片段。
$db = new swoole_mysql();
$config = array(
'host' => '127.0.0.1',
'port' => 3306,
'user' => 'test',
'password' => 'test',
'database' => 'test',
);
$db->connect($config, function ($db, $r) {
// 从 users 表中查询一条数据
$sql = 'select * from users where id = 1';
$db->query($sql, function(swoole_mysql $db, $r) {
if ($r !== false) {
// 查询成功后修改一条数据
$updateSql = 'update users set name="new name" where id = 1';
$db->query($updateSql, function (swoole_mysql $db, $r) {
$rows = $db->affected_rows;
if ($r === true) {
return $this->response->end('更新成功');
}
});
}
$db->close();
});
});
Copy to clipboardErrorCopied
注意
MySQL
等异步模块已在4.3.0中移除,并转移到了swoole_async。
从上面的代码片段可以看出,每一个操作几乎就需要一个回调函数,在复杂的业务场景中回调的层次感和代码结构绝对会让你崩溃,其实不难看出这样的写法有点类似 JavaScript
上的异步方法的写法,而 JavaScript
也为此提供了不少的解决方案(当然方案是源于其它编程语言),如 Promise
,yield + generator
, async/await
,Promise
则是对回调的一种封装方式,而 yield + generator
和 async/await
则需要在代码上显性的增加一些代码语法标记,这些相对比回调函数来说,不妨都是一些非常不错的解决方案,但是你需要另花时间来理解它的实现机制和语法。Swoole
协程也是对异步回调的一种解决方案,在 PHP
语言下,Swoole
协程与 yield + generator
都属于协程的解决方案,协程的解决方案可以使代码以近乎于同步代码的书写方式来书写异步代码,显性的区别则是 yield + generator
的协程机制下,每一处 I/O
操作的调用代码都需要在前面加上 yield
语法实现协程切换,每一层调用都需要加上,否则会出现意料之外的错误,而 Swoole
协程的解决方案对比于此就高明多了,在遇到 I/O
时底层自动的进行隐式协程切换,无需添加任何的额外语法,无需在代码前加上 yield
,协程切换的过程无声无息,极大的减轻了维护异步系统的心智负担。