PHP Conference Japan 2024

parallel\Events 類別

(0.9.0)

事件迴圈

事件迴圈會監控多個 futures 和/或 channels(目標)的狀態,以便在目標可用且操作可以在不阻塞事件迴圈的情況下執行時,執行讀取(parallel\Future::value()parallel\Channel::recv())和寫入(parallel\Channel::send())操作。

類別概要

final class parallel\Events implements Countable, Traversable {
/* 輸入 */
public setInput(Input $input): void
/* 目標 */
public addChannel(parallel\Channel $channel): void
public addFuture(string $name, parallel\Future $future): void
public remove(string $target): void
/* 行為 */
public setBlocking(bool $blocking): void
public setTimeout(int $timeout): void
/* 輪詢 */
}

目錄

新增註釋

使用者貢獻的註釋 3 則註釋

16
s dot laufer at homegear dot email
4 年前
<?php
/**
Example showing the usage of events.

The documentation still is very thin, so I'm not sure, the example is the best solution. But it works.
*/
use parallel\{Channel,Runtime,Events,Events\Event};

$myThread = function(Channel $channel) {
$events = new Events();
$events->addChannel($channel);
//$events->setBlocking(false); //Uncomment to don't block on Events::poll()
$events->setTimeout(1000000); //Comment when not blocking

while(true)
{
/*
...
Your code.
...
*/

//Read all available events
try
{
$event = NULL;
do
{
$event = $events->poll(); //Returns non-null if there is an event
if($event && $event->source == 'myChannel')
{
//It seems, the target gets deleted after returning an event,
//so add it again.
$events->addChannel($channel);
if(
$event->type == Event\Type::Read)
{
if(
is_array($event->value) && count($event->value) > 0)
{
if(
$event->value['name'] == 'stop')
{
echo
'Stopping thread';
return;
//Stop
}
else
{
echo
'Event: '.$event->value['name'].' => '.$event->value['value'].PHP_EOL;
}
}
}
else if(
$event->type == Event\Type::Close) return; //Stop
}
}
while(
$event);
}
catch(
Events\Error\Timeout $ex)
{
//Timeout
echo 'Timeout'.PHP_EOL;
}
}
};

class
MyClass {
private
$runtime;
private
$future;
private
$channel;

public function
start() {
//Create runtime
$this->runtime = new Runtime();

//Create buffered channel.
//Buffered channels don't block on Channel::send().
//Note that target names need to be unique within the process.
$this->channel = Channel::make('myChannel', Channel::Infinite);

global
$myThread;
$this->future = $this->runtime->run($myThread, [$this->channel]);
}

public function
stop() {
$this->channel->send(['name' => 'stop', 'value' => true]);

$this->future->value(); //Wait for thread to finish
$this->channel->close();
}

public function
emit(string $name, $value)
{
$this->channel->send(['name' => $name, 'value' => $value]);
}
}

$a = new MyClass();
$a->start();

for(
$i = 0; $i < 5; $i++)
{
$a->emit('test', $i);
sleep(0.5);
}

sleep(2);

for(
$i = 5; $i < 10; $i++)
{
$a->emit('test', $i);
sleep(0.5);
}

$a->stop();
?>
1
gam6itko
3 年前
<?php

// 這個例子展示了 Events 如何處理 Future 事件

use parallel\{Events, Events\Event, Runtime};

$events = new Events();

$runtime = new Runtime();

//讀取 (類型: 1)
$future = $runtime->run(
static function (
string $name) {
return
"Future#$name 結果";
},
[
'讀取']
);
$events->addFuture("Future#讀取", $future);

//取消 (類型: 4)
$future = $runtime->run(
static function (
string $name) {
throw new
\Exception("Exception#$name");
},
[
'取消']
);
$events->addFuture("Future#取消", $future);

//終止 (類型: 5)
$future = $runtime->run(
static function () {
sleep(100000);
},
[]
);
$events->addFuture("Future#終止", $future);
$future->cancel(); //終止它

//錯誤 (類型: 6)
$future = $runtime->run(
static function () {
$memoryEater = [];
$i = 0;
while (++
$i) {
$memoryEater[] = $i;
}
}
);
$events->addFuture("Future#錯誤", $future);

// 讀取事件

/** @var Event $event */
foreach ($events as $i => $event) {
echo
str_pad('', 50, '=') . " 事件_$i\n";
var_dump($event);
echo
"\n";
}
0
gam6itko
3 年前
<?php

// 以下範例顯示如何從主執行緒停止子執行緒

use parallel\{Channel, Events, Events\Event\Type, Runtime};

$fnThread = static function (Channel $channel) {
$events = new Events();
$events->addChannel($channel);
$events->setBlocking(false); // 不要在 Events::poll() 上阻塞

while (true) {
if (
$event = $events->poll()) {
//目標在返回事件後似乎會被刪除,所以再次添加它。
$events->addChannel($channel);

if (
Type::Read === $event->type) {
echo
"- 收到值: ".$event->value.PHP_EOL;
} elseif (
Type::Close === $event->type) {
echo
"- 收到關閉事件".PHP_EOL;
return
"我完成了";
}

}
echo
"-\n";
usleep(500_000);
}
};

// 主執行緒
$runtime = new Runtime();
$channel = new Channel();
$future = $runtime->run($fnThread, [$channel]);

$channel->send('message1');
sleep(2);
$channel->send('message2');
sleep(2);

echo
"關閉通道\n";
$channel->close();
echo
"future 回傳: ".$future->value();
echo
PHP_EOL;
To Top