2019-07-16 11:05:06 +08:00
|
|
|
|
# Task
|
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
现阶段 `Swoole` 暂时没有办法 `hook` 所有的阻塞函数,也就意味着有些函数仍然会导致 `进程阻塞`,从而影响协程的调度,此时我们可以通过使用 `Task` 组件来模拟协程处理,从而达到不阻塞进程调用阻塞函数的目的,本质上是仍是是多进程运行阻塞函数,所以性能上会明显地不如原生协程,具体取决于 `Task Worker` 的数量。
|
2019-07-16 11:05:06 +08:00
|
|
|
|
|
|
|
|
|
## 安装
|
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
```bash
|
2019-07-16 11:05:06 +08:00
|
|
|
|
composer require hyperf/task
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## 配置
|
|
|
|
|
|
|
|
|
|
因为 Task 并不是默认组件,所以在使用的时候需要在 `server.php` 增加 `Task` 相关的配置。
|
|
|
|
|
|
|
|
|
|
```php
|
|
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
|
|
use Hyperf\Server\SwooleEvent;
|
|
|
|
|
|
|
|
|
|
return [
|
2019-07-22 02:06:27 +08:00
|
|
|
|
// 这里省略了其它不相关的配置项
|
2019-07-16 11:05:06 +08:00
|
|
|
|
'settings' => [
|
2019-07-22 02:06:27 +08:00
|
|
|
|
// Task Worker 数量,根据您的服务器配置而配置适当的数量
|
|
|
|
|
'task_worker_num' => 8,
|
|
|
|
|
// 因为 `Task` 主要处理无法协程化的方法,所以这里推荐设为 `false`,避免协程下出现数据混淆的情况
|
|
|
|
|
'task_enable_coroutine' => false,
|
2019-07-16 11:05:06 +08:00
|
|
|
|
],
|
|
|
|
|
'callbacks' => [
|
2019-07-22 02:06:27 +08:00
|
|
|
|
// Task callbacks
|
2019-07-16 11:05:06 +08:00
|
|
|
|
SwooleEvent::ON_TASK => [Hyperf\Framework\Bootstrap\TaskCallback::class, 'onTask'],
|
|
|
|
|
SwooleEvent::ON_FINISH => [Hyperf\Framework\Bootstrap\FinishCallback::class, 'onFinish'],
|
|
|
|
|
],
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
```
|
2019-07-24 13:42:59 +08:00
|
|
|
|
|
2019-07-16 11:05:06 +08:00
|
|
|
|
## 使用
|
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
Task 组件提供了 `主动方法投递` 和 `注解投递` 两种使用方法。
|
2019-07-16 11:05:06 +08:00
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
### 主动方法投递
|
2019-07-16 11:05:06 +08:00
|
|
|
|
|
|
|
|
|
```php
|
|
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
use Hyperf\Utils\Coroutine;
|
|
|
|
|
use Hyperf\Utils\ApplicationContext;
|
|
|
|
|
use Hyperf\Task\TaskExecutor;
|
|
|
|
|
use Hyperf\Task\Task;
|
|
|
|
|
|
|
|
|
|
class MethodTask
|
|
|
|
|
{
|
|
|
|
|
public function handle($cid)
|
|
|
|
|
{
|
|
|
|
|
return [
|
|
|
|
|
'worker.cid' => $cid,
|
2019-07-22 02:06:27 +08:00
|
|
|
|
// task_enable_coroutine 为 false 时返回 -1,反之 返回对应的协程 ID
|
2019-07-16 11:05:06 +08:00
|
|
|
|
'task.cid' => Coroutine::id(),
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$container = ApplicationContext::getContainer();
|
|
|
|
|
$exec = $container->get(TaskExecutor::class);
|
2019-07-31 16:26:40 +08:00
|
|
|
|
$result = $exec->execute(new Task([MethodTask::class, 'handle'], [Coroutine::id()]));
|
2019-07-16 11:05:06 +08:00
|
|
|
|
|
|
|
|
|
```
|
2019-07-24 13:42:59 +08:00
|
|
|
|
|
2019-07-16 11:05:06 +08:00
|
|
|
|
### 使用注解
|
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
通过 `主动方法投递` 时,并不是特别直观,这里我们实现了对应的 `@Task` 注解,并通过 `AOP` 重写了方法调用。当在 `Worker` 进程时,自动投递到 `Task` 进程,并协程等待 数据返回。
|
2019-07-16 11:05:06 +08:00
|
|
|
|
|
|
|
|
|
```php
|
|
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
use Hyperf\Utils\Coroutine;
|
|
|
|
|
use Hyperf\Utils\ApplicationContext;
|
|
|
|
|
use Hyperf\Task\Annotation\Task;
|
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
class AnnotationTask
|
2019-07-16 11:05:06 +08:00
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @Task
|
|
|
|
|
*/
|
|
|
|
|
public function handle($cid)
|
|
|
|
|
{
|
|
|
|
|
return [
|
|
|
|
|
'worker.cid' => $cid,
|
|
|
|
|
// task_enable_coroutine=false 时返回 -1,反之 返回对应的协程 ID
|
|
|
|
|
'task.cid' => Coroutine::id(),
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$container = ApplicationContext::getContainer();
|
2019-07-22 02:06:27 +08:00
|
|
|
|
$task = $container->get(AnnotationTask::class);
|
2019-07-16 11:05:06 +08:00
|
|
|
|
$result = $task->handle(Coroutine::id());
|
|
|
|
|
```
|
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
> 使用 `@Task` 注解时需 `use Hyperf\Task\Annotation\Task;`
|
|
|
|
|
|
2019-07-16 11:05:06 +08:00
|
|
|
|
## 附录
|
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
Swoole 暂时没有协程化的函数列表
|
2019-07-16 11:05:06 +08:00
|
|
|
|
|
2019-07-22 02:06:27 +08:00
|
|
|
|
- mysql:底层使用 libmysqlclient
|
|
|
|
|
- curl:底层使用 libcurl(即不能使用CURL驱动的Guzzle)
|
|
|
|
|
- mongo:底层使用 mongo-c-client
|
2019-07-16 11:05:06 +08:00
|
|
|
|
- pdo_pgsql
|
|
|
|
|
- pdo_ori
|
|
|
|
|
- pdo_odbc
|
|
|
|
|
- pdo_firebird
|
|
|
|
|
|
2019-07-24 13:42:59 +08:00
|
|
|
|
### MongoDB
|
|
|
|
|
|
2019-07-24 13:48:17 +08:00
|
|
|
|
> 因为 `MongoDB` 没有办法被 `hook`,所以我们可以通过 `Task` 来调用,下面就简单介绍一下如何通过注解方式调用 `MongoDB`。
|
2019-07-24 13:42:59 +08:00
|
|
|
|
|
|
|
|
|
以下我们实现两个方法 `insert` 和 `query`,其中需要注意的是 `manager` 方法不能使用 `Task`,
|
2019-07-24 13:48:17 +08:00
|
|
|
|
因为 `Task` 会在对应的 `Task进程` 中处理,然后将数据从 `Task进程` 返回到 `Worker进程` 。
|
|
|
|
|
所以 `Task方法` 的入参和出参最好不要携带任何 `IO`,比如返回一个实例化后的 `Redis` 等等。
|
2019-07-24 13:42:59 +08:00
|
|
|
|
|
|
|
|
|
```php
|
|
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
|
|
namespace App\Task;
|
|
|
|
|
|
|
|
|
|
use Hyperf\Task\Annotation\Task;
|
|
|
|
|
use MongoDB\Driver\BulkWrite;
|
|
|
|
|
use MongoDB\Driver\Manager;
|
|
|
|
|
use MongoDB\Driver\Query;
|
|
|
|
|
use MongoDB\Driver\WriteConcern;
|
|
|
|
|
|
|
|
|
|
class MongoTask
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @var Manager
|
|
|
|
|
*/
|
|
|
|
|
public $manager;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @Task
|
2019-07-24 13:48:17 +08:00
|
|
|
|
* @param string $namespace
|
|
|
|
|
* @param array $document
|
2019-07-24 13:42:59 +08:00
|
|
|
|
*/
|
|
|
|
|
public function insert($namespace, $document)
|
|
|
|
|
{
|
|
|
|
|
$writeConcern = new WriteConcern(WriteConcern::MAJORITY, 1000);
|
|
|
|
|
$bulk = new BulkWrite();
|
|
|
|
|
$bulk->insert($document);
|
|
|
|
|
|
|
|
|
|
$result = $this->manager()->executeBulkWrite($namespace, $bulk, $writeConcern);
|
|
|
|
|
return $result->getUpsertedCount();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @Task
|
2019-07-24 13:48:17 +08:00
|
|
|
|
* @param string $namespace
|
|
|
|
|
* @param array $filter
|
|
|
|
|
* @param array $options
|
2019-07-24 13:42:59 +08:00
|
|
|
|
*/
|
|
|
|
|
public function query($namespace, $filter = [], $options = [])
|
|
|
|
|
{
|
|
|
|
|
$query = new Query($filter, $options);
|
|
|
|
|
$cursor = $this->manager()->executeQuery($namespace, $query);
|
|
|
|
|
return $cursor->toArray();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected function manager()
|
|
|
|
|
{
|
|
|
|
|
if ($this->manager instanceof Manager) {
|
|
|
|
|
return $this->manager;
|
|
|
|
|
}
|
|
|
|
|
$uri = 'mongodb://127.0.0.1:27017';
|
|
|
|
|
return $this->manager = new Manager($uri, []);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
使用如下
|
|
|
|
|
|
|
|
|
|
```php
|
|
|
|
|
<?php
|
|
|
|
|
use App\Task\MongoTask;
|
|
|
|
|
use Hyperf\Utils\ApplicationContext;
|
|
|
|
|
|
|
|
|
|
$client = ApplicationContext::getContainer()->get(MongoTask::class);
|
|
|
|
|
$client->insert('hyperf.test', ['id' => rand(0, 99999999)]);
|
|
|
|
|
|
|
|
|
|
$result = $client->query('hyperf.test', [], [
|
|
|
|
|
'sort' => ['id' => -1],
|
|
|
|
|
'limit' => 5,
|
|
|
|
|
]);
|
|
|
|
|
```
|
|
|
|
|
|