协程:实现 sync.WaitGroup 功能
最后更新于:2022-04-02 06:16:36
# 协程:实现 sync.WaitGroup 功能
[TOC]
在`Swoole4`中可以使用`channel`实现协程间的通信、依赖管理、协程同步。基于`channel`可以很容易地实现`Golang`的`sync.WaitGrup`功能。
## 实现代码
~~~
class WaitGroup
{
private $count = 0;
private $chan;
/**
* waitgroup constructor.
* @desc 初始化一个channel
*/
public function __construct()
{
$this->chan = new chan;
}
public function add()
{
$this->count++;
}
public function done()
{
$this->chan->push(true);
}
public function wait()
{
while($this->count--)
{
$this->chan->pop();
}
}
}
~~~
* `add`方法增加计数
* `done`表示任务已完成
* `wait`等待所有任务完成恢复当前协程的执行
* `WaitGroup`对象可以复用,`add`、`done`、`wait`之后可以再次使用
## 使用实例
~~~
go(function () {
$wg = new waitgroup();
$result = [];
$wg->add();
//启动第一个协程
go(function () use ($wg, &$result) {
//启动一个协程客户端client,请求淘宝首页
$cli = new Client('www.taobao.com', 443, true);
$cli->setHeaders([
'Host' => "www.taobao.com",
"User-Agent" => 'Chrome/49.0.2587.3',
'Accept' => 'text/html,application/xhtml+xml,application/xml',
'Accept-Encoding' => 'gzip',
]);
$cli->set(['timeout' => 1]);
$cli->get('/index.php');
$result['taobao'] = $cli->body;
$cli->close();
$wg->done();
});
$wg->add();
//启动第二个协程
go(function () use ($wg, &$result) {
//启动一个协程客户端client,请求百度首页
$cli = new Client('www.baidu.com', 443, true);
$cli->setHeaders([
'Host' => "www.baidu.com",
"User-Agent" => 'Chrome/49.0.2587.3',
'Accept' => 'text/html,application/xhtml+xml,application/xml',
'Accept-Encoding' => 'gzip',
]);
$cli->set(['timeout' => 1]);
$cli->get('/index.php');
$result['baidu'] = $cli->body;
$cli->close();
$wg->done();
});
//挂起当前协程,等待所有任务完成后恢复
$wg->wait();
//这里 $result 包含了 2 个任务执行结果
var_dump($result);
});
~~~
';