现在的PHPer,得益于前端的快速发展、以及前后端分离的理念普及,已经有很多人都从渲染模板的工作中解脱出来了。
随着微服务(服务化)的盛行,让swoole如虎添翼,php不会越来越衰颓,swoole将撑起php的另一片天。
随着PHP编程语言被越来越多的人使用,更多人了解到swoole的存在,有的人说swoole是披着PHP外衣的C程序,但是swoole并不是一个简单的PHP框架,正如swoole官方首页的第一句话“重新定义PHP”,千万不要用旧有php的思想来写swoole代码!
Swoole的发展和完善,也促成了大量PHPer的自我提升,swoole重新激活了PHP,php成就了swoole!
一、什么是Swoole?
Swoole 是“面向生产环境的 PHP 异步网络通信引擎”,首先 Swoole 它是一个网络应用的开发工具,它支持 Http、TCP、UDP、WebSocket,它的使用目前一些基于 Swoole 的框架开发,从开发习惯上和传统的TP、LV 框架相差不多。
二、Swoole给PHP带来了什么?
1、类NodeJS的异步:
备胎倒不是没有,最有名的当属Workerman和ReactPHP,两者都提供了类NodeJS的异步能力,还是用PHP实现的,相比Swoole显著降低了内存泄露和崩溃的几率。
当异步回调面临复杂业务逻辑的时候,回调写在嵌套和循环里就不是那么直观了,不但需要自己保存context,有时不得不写成递归的形式,如果要在callback chain中跳来跳去就更烧脑了。Swoole协程彻底解决了复杂业务时的异步回调地狱。这点没有替代方案。
2、类Go的协程
Swoole 的协程客户端必须在协程的上下文环境中使用。
// 第一种情况:Request 回调本身是协程环境
$server->on('Request', function($request, $response) {
// 创建 Mysql 协程客户端
$mysql = new Swoole\Coroutine\MySQL();
$mysql->connect([]);
$mysql->query();
});
// 第二种情况:WorkerStart 回调不是协程环境
$server->on('WorkerStart', function() {
// 需要先声明一个协程环境,才能使用协程客户端
go(function(){
// 创建 Mysql 协程客户端
$mysql = new Swoole\Coroutine\MySQL();
$mysql->connect([]);
$mysql->query();
});
});
复制代码
Swoole 的协程是基于单线程的, 无法利用多核CPU,同一时间只有一个在调度。
// 启动 4 个协程
$n = 4;
for ($i = 0; $i < $n; $i++) {
go(function () use ($i) {
// 模拟 IO 等待
Co::sleep(1);
echo microtime(true) . ": hello $i " . PHP_EOL;
});
};
echo "hello main \n";
// 每次输出的结果都是一样
$ php test.php
hello main
1558749158.0913: hello 0
1558749158.0915: hello 3
1558749158.0915: hello 2
1558749158.0915: hello 1
复制代码
Swoole 协程使用示例及详解
// 创建一个 Http 服务
$server = new Swoole\Http\Server('127.0.0.1', 9501, SWOOLE_BASE);
// 调用 onRequest 事件回调函数时,底层会调用 C 函数 coro_create 创建一个协程,
// 同时保存这个时间点的 CPU 寄存器状态和 ZendVM stack 信息。
$server->on('Request', function($request, $response) {
// 创建一个 Mysql 的协程客户端
$mysql = new Swoole\Coroutine\MySQL();
// 调用 mysql->connect 时发生 IO 操作,底层会调用 C 函数 coro_save 保存当前协程的状态,
// 包括 Zend VM 上下文以及协程描述的信息,并调用 coro_yield 让出程序控制权,当前的请求会挂起。
// 当协程让出控制权之后,会继续进入 EventLoop 处理其他事件,这时 Swoole 会继续去处理其他客户端发来的 Request。
$res = $mysql->connect([
'host' => '127.0.0.1',
'user' => 'root',
'password' => 'root',
'database' => 'test'
]);
// IO 事件完成后,MySQL 连接成功或失败,底层调用 C 函数 coro_resume 恢复对应的协程,恢复 ZendVM 上下文,继续向下执行 PHP 代码。
if ($res == false) {
$response->end("MySQL connect fail");
return;
}
// mysql->query 的执行过程和 mysql->connect 一致,也会进行一次协程切换调度
$ret = $mysql->query('show tables', 2);
// 所有操作完成后,调用 end 方法返回结果,并销毁此协程。
$response->end('swoole response is ok, result='.var_export($ret, true));
});
// 启动服务
$server->start();
复制代码
3、内置Redis/Http等协议
内置协议,Swoole支持的协议最多最全,即使不支持的,也有很大几率使用HOOK自动支持起来。
4、常驻内存,避免重复加载带来的性能损耗,提升海量性能
5、现有阻塞库异步/协程化,提高对IO密集型场景并发处理能力(如:微信开发、支付、登录等)。
现有阻塞库的异步/协程化,这点也没有替代方案,Workerman和ReactPHP里非要用阻塞库的时候,只能走多线程或者多进程方案,编程不知道要麻烦多少。
三、使用swoole必须要掌握的技能
1、多线程编程
2、进程间通信
3、网络协议TCP/UDP的认知
4、PHP的各项基本技能