php为什么要使用swoole?

现在的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的各项基本技能

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章