Swoole4
为PHP
语言提供了强大的CSP
协程编程模式。底层提供了3
个关键词,能够方便地实现各种功能。php
Swoole4
提供的PHP协程
语法借鉴自Golang
,在此向GO
开发组致敬PHP+Swoole
协程能够与Golang
很好地互补。Golang
:静态语言,严谨强大性能好,PHP+Swoole
:动态语言,灵活简单易用本文基于Swoole-4.2.9
和PHP-7.2.9
版本
go
:建立一个协程chan
:建立一个通道defer
:延迟任务,在协程退出时执行,先进后出这3
个功能底层实现所有为内存操做,没有任何IO
资源消耗。就像PHP
的Array
同样是很是廉价的。若是有须要就能够直接使用。这与socket
和file
操做不一样,后者须要向操做系统申请端口和文件描述符,读写可能会产生阻塞的IO
等待。html
使用go
函数可让一个函数并发地去执行。在编程过程当中,若是某一段逻辑能够并发执行,就能够将它放置到go
协程中执行。mysql
function test1() { sleep(1); echo "b"; } function test2() { sleep(2); echo "c"; } test1(); test2();
htf@LAPTOP-0K15EFQI:~$ time php b1.php bc real 0m3.080s user 0m0.016s sys 0m0.063s htf@LAPTOP-0K15EFQI:~$
上述代码中,test1
和test2
会顺序执行,须要3
秒才能执行完成。redis
使用go
建立协程,可让test1
和test2
两个函数变成并发执行。sql
Swoole\Runtime::enableCoroutine(); go(function () { sleep(1); echo "b"; }); go(function () { sleep(2); echo "c"; });
Swoole\Runtime::enableCoroutine()
做用是将PHP
提供的stream
、sleep
、pdo
、mysqli
、redis
等功能从同步阻塞切换为协程的异步IO
bchtf@LAPTOP-0K15EFQI:~$ time php co.php bc real 0m2.076s user 0m0.000s sys 0m0.078s htf@LAPTOP-0K15EFQI:~$
能够看到这里只用了2
秒就执行完成了。shell
t1+t2+t3...
max(t1, t2, t3, ...)
有了go
关键词以后,并发编程就简单多了。与此同时又带来了新问题,若是有2
个协程并发执行,另一个协程,须要依赖这两个协程的执行结果,若是解决此问题呢?编程
答案就是使用通道(Channel
),在Swoole4
协程中使用new chan
就能够建立一个通道。通道能够理解为自带协程调度的队列。它有两个接口push
和pop
:swoole
push
:向通道中写入内容,若是已满,它会进入等待状态,有空间时自动恢复pop
:从通道中读取内容,若是为空,它会进入等待状态,有数据时自动恢复使用通道能够很方便地实现并发管理。并发
$chan = new chan(2); # 协程1 go (function () use ($chan) { $result = []; for ($i = 0; $i < 2; $i++) { $result += $chan->pop(); } var_dump($result); }); # 协程2 go(function () use ($chan) { $cli = new Swoole\Coroutine\Http\Client('www.qq.com', 80); $cli->set(['timeout' => 10]); $cli->setHeaders([ 'Host' => "www.qq.com", "User-Agent" => 'Chrome/49.0.2587.3', 'Accept' => 'text/html,application/xhtml+xml,application/xml', 'Accept-Encoding' => 'gzip', ]); $ret = $cli->get('/'); // $cli->body 响应内容过大,这里用 Http 状态码做为测试 $chan->push(['www.qq.com' => $cli->statusCode]); }); # 协程3 go(function () use ($chan) { $cli = new Swoole\Coroutine\Http\Client('www.163.com', 80); $cli->set(['timeout' => 10]); $cli->setHeaders([ 'Host' => "www.163.com", "User-Agent" => 'Chrome/49.0.2587.3', 'Accept' => 'text/html,application/xhtml+xml,application/xml', 'Accept-Encoding' => 'gzip', ]); $ret = $cli->get('/'); // $cli->body 响应内容过大,这里用 Http 状态码做为测试 $chan->push(['www.163.com' => $cli->statusCode]); });
htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$ time php co2.php array(2) { ["www.qq.com"]=> int(302) ["www.163.com"]=> int(200) } real 0m0.268s user 0m0.016s sys 0m0.109s htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$
这里使用go
建立了3
个协程,协程2
和协程3
分别请求qq.com
和163.com
主页。协程1
须要拿到Http
请求的结果。这里使用了chan
来实现并发管理。app
1
循环两次对通道进行pop
,由于队列为空,它会进入等待状态2
和协程3
执行完成后,会push
数据,协程1
拿到告终果,继续向下执行在协程编程中,可能须要在协程退出时自动实行一些任务,作清理工做。相似于PHP
的register_shutdown_function
,在Swoole4
中可使用defer
实现。
Swoole\Runtime::enableCoroutine(); go(function () { echo "a"; defer(function () { echo "~a"; }); echo "b"; defer(function () { echo "~b"; }); sleep(1); echo "c"; });
htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$ time php defer.php abc~b~a real 0m1.068s user 0m0.016s sys 0m0.047s htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$
Swoole4
提供的Go + Chan + Defer
为PHP
带来了一种全新的CSP
并发编程模式。灵活使用Swoole4
提供的各项特性,能够解决工做中各种复杂功能的设计和开发。