hyperf/docs/zh-hk/process.md

183 lines
4.3 KiB
Markdown
Raw Normal View History

2019-12-12 16:24:04 +08:00
# 自定義進程
2019-04-05 09:35:27 +08:00
2019-12-12 16:24:04 +08:00
[hyperf/process](https://github.com/hyperf/process) 可以添加一個用户自定義的工作進程,此函數通常用於創建一個特殊的工作進程,用於監控、上報或者其他特殊的任務。在 Server 啟動時會自動創建進程並執行指定的子進程函數進程意外退出時Server 會重新拉起進程。
2019-04-05 09:35:27 +08:00
2019-12-12 16:24:04 +08:00
## 創建一個自定義進程
2019-04-05 09:35:27 +08:00
2019-12-12 16:24:04 +08:00
在任意位置實現一個繼承 `Hyperf\Process\AbstractProcess` 的子類,並實現接口方法 `handle(): void`,方法內實現您的邏輯代碼,我們通過代碼來舉例:
2019-05-07 17:46:12 +08:00
```php
<?php
declare(strict_types=1);
namespace App\Process;
use Hyperf\Process\AbstractProcess;
class FooProcess extends AbstractProcess
{
public function handle(): void
{
2019-12-12 16:24:04 +08:00
// 您的代碼 ...
2019-05-07 17:46:12 +08:00
}
}
```
2019-12-12 16:24:04 +08:00
這樣即完成了一個自定義進程類,但該自定義進程類尚未被註冊到 `進程管理器(ProcessManager)` 內,我們可以通過 `配置文件``註解` 兩種方式的任意一種來完成註冊工作。
2019-05-07 17:46:12 +08:00
2019-12-12 16:24:04 +08:00
### 通過配置文件註冊
2019-05-07 17:46:12 +08:00
2019-12-12 16:24:04 +08:00
只需在 `config/autoload/processes.php` 內加上您的自定義進程類即可:
2019-05-07 17:46:12 +08:00
```php
// config/autoload/processes.php
return [
\App\Process\FooProcess::class,
];
```
2019-12-12 16:24:04 +08:00
### 通過註解註冊
2019-05-07 17:46:12 +08:00
2019-12-12 16:24:04 +08:00
只需在自定義進程類上定義 `@Process` 註解Hyperf 會收集並自動完成註冊工作:
2019-05-07 17:46:12 +08:00
```php
<?php
declare(strict_types=1);
namespace App\Process;
use Hyperf\Process\AbstractProcess;
use Hyperf\Process\Annotation\Process;
/**
* @Process(name="foo_process")
*/
class FooProcess extends AbstractProcess
{
public function handle(): void
{
2019-12-12 16:24:04 +08:00
// 您的代碼 ...
2019-05-07 17:46:12 +08:00
}
}
```
2019-12-12 16:24:04 +08:00
> 使用 `@Process` 註解時需 `use Hyperf\Process\Annotation\Process;` 命名空間;
2019-05-07 17:46:12 +08:00
2019-12-12 16:24:04 +08:00
## 為進程啟動加上條件
2019-05-07 17:46:12 +08:00
2019-12-12 16:24:04 +08:00
有些時候,並不是所有時候都應該啟動一個自定義進程,一個自定義進程的啟動與否可能會根據某些配置或者某些條件來決定,我們可以通過在自定義進程類內重寫 `isEnable(): bool` 方法來實現,默認返回 `true`,即會跟隨服務一同啟動,如方法返回 `false`,則服務啟動時不會啟動該自定義進程。
2019-05-07 17:46:12 +08:00
```php
<?php
declare(strict_types=1);
namespace App\Process;
use Hyperf\Process\AbstractProcess;
use Hyperf\Process\Annotation\Process;
/**
* @Process(name="foo_process")
*/
class FooProcess extends AbstractProcess
{
public function handle(): void
{
2019-12-12 16:24:04 +08:00
// 您的代碼 ...
2019-05-07 17:46:12 +08:00
}
public function isEnable($server): bool
2019-05-07 17:46:12 +08:00
{
2019-12-12 16:24:04 +08:00
// 不跟隨服務啟動一同啟動
2019-05-07 17:46:12 +08:00
return false;
}
}
```
2019-12-12 16:24:04 +08:00
## 設置自定義進程
2019-05-07 17:46:12 +08:00
2019-12-12 16:24:04 +08:00
自定義進程存在一些可設置的參數,均可以通過 在子類上重寫參數對應的屬性 或 在 `@Process` 註解內定義對應的屬性 兩種方式來進行定義。
2019-04-05 09:35:27 +08:00
```php
<?php
2019-05-07 17:46:12 +08:00
declare(strict_types=1);
2019-04-05 09:35:27 +08:00
2019-05-07 17:46:12 +08:00
namespace App\Process;
use Hyperf\Process\AbstractProcess;
use Hyperf\Process\Annotation\Process;
/**
2021-03-29 09:30:55 +08:00
* @Process(name="user-process"redirectStdinStdout=false, pipeType=2, enableCoroutine=true)
2019-05-07 17:46:12 +08:00
*/
class FooProcess extends AbstractProcess
{
/**
2019-12-12 16:24:04 +08:00
* 進程數量
2019-05-07 17:46:12 +08:00
* @var int
*/
public $nums = 1;
/**
2019-12-12 16:24:04 +08:00
* 進程名稱
2019-05-07 17:46:12 +08:00
* @var string
*/
public $name = 'user-process';
/**
2019-12-12 16:24:04 +08:00
* 重定向自定義進程的標準輸入和輸出
2019-05-07 17:46:12 +08:00
* @var bool
*/
public $redirectStdinStdout = false;
/**
2019-12-12 16:24:04 +08:00
* 管道類型
2019-05-07 17:46:12 +08:00
* @var int
*/
public $pipeType = 2;
/**
2019-12-12 16:24:04 +08:00
* 是否啟用協程
2019-05-07 17:46:12 +08:00
* @var bool
*/
public $enableCoroutine = true;
}
```
## 使用示例
2020-03-10 11:41:31 +08:00
我們創建一個用於監控失敗隊列數量的子進程,當失敗隊列有數據時,報出警告。
2019-05-07 17:46:12 +08:00
```php
<?php
2019-04-05 09:35:27 +08:00
declare(strict_types=1);
2019-04-05 09:45:00 +08:00
namespace App\Process;
2019-04-05 09:35:27 +08:00
use Hyperf\Process\AbstractProcess;
use Hyperf\Process\Annotation\Process;
use Hyperf\Contract\StdoutLoggerInterface;
/**
* @Process(name="demo_process")
*/
class DemoProcess extends AbstractProcess
{
public function handle(): void
{
$logger = $this->container->get(StdoutLoggerInterface::class);
while (true) {
$redis = $this->container->get(\Redis::class);
$count = $redis->llen('queue:failed');
if ($count > 0) {
$logger->warning('The num of failed queue is ' . $count);
}
sleep(1);
}
}
}
2021-03-29 09:30:55 +08:00
```