すべてのコンピューターには少なくとも2つのコアが必要ですが、コンピューターで実行されているプログラムのほとんどは、CPUのコアの数をはるかに超えています。これは、オペレーティング システムがタスク処理においてマクロ並列およびマイクロシリアル アプローチを採用しているためです。つまり、CPU は各プログラムをしばらく実行した後、別のプログラムの実行に切り替えます。誰もがたくさんのパフォーマンスをしたように見えます。現在はphp8.1です。ファイバーを導入しました。スケジュール権限はすべての PHP 開発者に与えられます。それでは、ファイバーを使用してどのような新しい操作を実装できるでしょうか? (この記事はすべての人へのヒントです。さらに興味深い使用方法を追加することは歓迎です)
例として、誰もが通常作成する for ループを考えてみましょう。 go のように go
を 2 つ書き、それぞれに同時に入力するループを書くと、交互に出力されることがわかります。過去の PHP バージョンでは、CLI を 1 つだけ開いて複数の for ループを作成した場合、その出力はシーケンシャルである必要があります。クロス出力は実現できません (つまり、最初のループで数回実行した後、再度 b を実行でき、一定期間 b を実行した後、A を実行できます)。
ファイバーの助けを借りて、この操作も実現できます。 [推奨される学習: PHP ビデオ チュートリアル ]
次のコードは、クロス実行の 2 つのループを実現できます。 2 つのプログラムの実行頻度も制御できます (たとえば、A は 3 回実行され、B は 1 回実行されます)
<?php $t1 = false; $t2 = false; $reg = []; $reg[] = new \Fiber(function () use (&$t1) { for ($i = 1; $i < 10; $i++) { echo $i; echo PHP_EOL; \Fiber::suspend(); } $t1 = true; }); $reg[] = new \Fiber(function () use (&$t2) { for ($i = 1; $i < 10; $i++) { echo $i; echo PHP_EOL; \Fiber::suspend(); } $t2 = true; }); $startTag = true; while (count($reg) > 1) { if ($startTag) foreach ($reg as $pI) { $pI->start(); $startTag = false; } foreach ($reg as $pI) { $pI->resume(); } if ($t1 === true && $t2 === true) { break; } }
1 1 2 2 3 3 4 4 5 5 6 6 7 7 8 8 9 9
たとえば、最初のループの後の 2 つのループの実行頻度も制御できます。ループは 3 回実行され、2 番目のループは 1 回実行されます。コードは次のとおりです
<?php $reg = []; $fId = 1; $reg[$fId] = new \Fiber(function () use (&$reg, $fId) { for ($i = 1; $i < 10; $i++) { echo $fId . ':' . $i; echo PHP_EOL; if ($i % 3 == 0) { \Fiber::suspend(); } } unset($reg[$fId]); }); $fId++; $reg[$fId] = new \Fiber(function () use (&$reg, $fId) { for ($i = 1; $i < 10; $i++) { echo $fId . ':' . $i; echo PHP_EOL; \Fiber::suspend(); } unset($reg[$fId]); }); $startTag = true; while (count($reg) > 0) { if ($startTag) foreach ($reg as $pI) { $pI->start(); $startTag = false; } foreach ($reg as $pI) { $pI->resume(); } }
1:1 1:2 1:3 2:1 1:4 1:5 1:6 2:2 1:7 1:8 1:9 2:3 2:4 2:5 2:6 2:7 2:8 2:9
メッセージ通知を通じて完了
<?php namespace App\Command; use Symfony\Component\Console\Attribute\AsCommand; use Symfony\Component\Console\Command\Command; use Symfony\Component\Console\Input\InputArgument; use Symfony\Component\Console\Input\InputInterface; use Symfony\Component\Console\Input\InputOption; use Symfony\Component\Console\Output\OutputInterface; use Symfony\Component\Console\Style\SymfonyStyle; #[AsCommand( name: 'Sname', description: 'Add a short description for your command', )] class SnameCommand extends Command { protected function configure(): void { $this ->addArgument('arg1', InputArgument::OPTIONAL, 'Argument description') ->addOption('option1', null, InputOption::VALUE_NONE, 'Option description'); } protected function execute(InputInterface $input, OutputInterface $output): int { $t1 = false; $t2 = false; $reg = []; $fId = 1; $reg[] = new \Fiber(function () use ($fId) { for ($i = 1; $i < 10; $i++) { echo $fId . ':' . $i; echo PHP_EOL; if ($i % 3 == 0) { \Fiber::suspend(new SuspendData(Status::Running)); } } \Fiber::suspend(new SuspendData(Status::Stop)); }); $fId++; $reg[] = new \Fiber(function () use ($fId) { for ($i = 1; $i < 10; $i++) { echo $fId . ':' . $i; echo PHP_EOL; \Fiber::suspend(new SuspendData(Status::Running)); } \Fiber::suspend(new SuspendData(Status::Stop)); }); $startTag = true; while (count($reg) > 0) { if ($startTag) foreach ($reg as $pI) { $pI->start(); $startTag = false; } foreach ($reg as $key => $pI) { $r = $pI->resume(); if ($r->status === Status::Stop) { unset($reg[$key]); } } } return Command::SUCCESS; } } class SuspendData { public readonly Status $status; public function __construct($status) { $this->status = $status; } } enum Status { case Stop; case Running; }
以上がPHP8.1 Fiber クロス実行マルチタスク (詳細なコード説明付き)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。