内容简介:这使用
Swoole4
提供了强大的 PHP CSP
协程编程模式。底层提供了 3
个关键词,可以方便地实现各类功能。
关键词
go chan defer
这 3
个功能底层实现全部为 内存操作
,没有任何 IO
资源消耗。就像 PHP
的 Array
一样是非常廉价的。如果有需要就可以直接使用。这与 socket
和 file
操作不同,后者需要向操作系统申请端口和文件描述符,读写可能会产生阻塞的 IO
等待。
协程并发
使用 go
函数可以让一个函数并发地去执行。在编程过程中,如果某一段逻辑可以并发执行,就可以将它放置到 go
协程中执行。
顺序执行
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
秒才能执行完成。
并发执行
使用 go
创建协程,可以让 test1
和 test2
两个函数变成并发执行。
Swoole\Runtime::enableCoroutine();
go(function ()
{
sleep(1);
echo "b";
});
go(function ()
{
sleep(2);
echo "c";
});
执行结果:
bchtf@LAPTOP-0K15EFQI:~$ time php co.php bc real 0m2.076s user 0m0.000s sys 0m0.078s htf@LAPTOP-0K15EFQI:~$
可以看到这里只用了 2
秒就执行完成了。
t1+t2+t3... max(t1, t2, t3, ...)
协程通信
有了 go
关键词之后,并发编程就简单多了。与此同时又带来了新问题,如果有 2
个协程并发执行,另外一个协程,需要依赖这两个协程的执行结果,如果解决此问题呢?
答案就是使用通道( Channel
),在 Swoole4
协程中使用 new chan
就可以创建一个通道。通道可以理解为自带协程调度的队列。它有两个接口 push
和 pop
:
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
来实现并发管理。
-
协程
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
提供的各项特性,可以解决工作中各类复杂功能的设计和开发。
以上所述就是小编给大家介绍的《PHP 协程:Go + Chan + Defer》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Pro CSS and HTML Design Patterns
Michael Bowers / Apress / April 23, 2007 / $44.99
Design patterns have been used with great success in software programming. They improve productivity, creativity, and efficiency in web design and development, and they reduce code bloat and complexit......一起来看看 《Pro CSS and HTML Design Patterns》 这本书的介绍吧!