Coroutine\Channel::select

废弃

  1. 此方法存在安全问题,已废弃

通道读写检测。类似于socket_selectstream_select可以检测channel是否可进行读写。

原型

  1. function Coroutine\Channel::select(array &$read, array &$write, float $timeout = -1);

$read$write数组中有部分channel对象处于可读或可写状态,select会立即返回,不会产生协程调度。当数组中没有任何channel可读或可写时,将挂起当前协程,并设置定时器。当其中一个通道可读或可写时,将重新唤醒当前协程。

select操作只检测channel列表的可读或可写状态,但并不会读写channel,在select调用返回后,可遍历$read$write数组,执行poppush方法,完成通道读写操作。

参数

  • $read 数组引用类型,元素为channel对象,读操作检测,可以为null
  • $write 数组引用类型,元素为channel对象,写操作检测,可以为null
  • $timeout 浮点型,超时设置,单位为秒,最小粒度为0.001秒,即1ms。默认为0,表示永不超时。

返回值

  • 成功返回true,底层会修改$read$write数组,$read$write中的元素,即是可读或可写的channel
  • 超时或传入的参数错误,如$read$write中有非channel对象,底层返回false

注意事项

早期版本中Coroutine\Channel由于存在一些问题,在Swoole4.0.3版本重构并废弃了 Coroutine\Channel::select 方法。所以在Swoole4.0.3以上的版本请使用channel->pop($timeout)替代。

fibonacci 实例

  1. $c1 = new chan();
  2. $c2 = new chan();
  3. function fibonacci($c1, $c2)
  4. {
  5. go(function () use ($c1, $c2) {
  6. $a = 0;
  7. $b = 1;
  8. while(1) {
  9. $read_list = [$c2];
  10. $write_list = [$c1];
  11. $result = chan::select($read_list, $write_list, 2);
  12. if ($write_list) {
  13. $t = $a + $b;
  14. $a = $b;
  15. $b = $t;
  16. $c1->push($a);
  17. }
  18. if ($read_list) {
  19. $ret = $c2->pop();
  20. if ($ret === 1) {
  21. return 1;
  22. }
  23. }
  24. }
  25. });
  26. }
  27. $num = 10;
  28. go(function () use ($c1, $c2, $num) {
  29. for ($i = 0; $i < $num; $i ++) {
  30. $ret = $c1->pop();
  31. echo "fibonacci @$i $ret\n";
  32. }
  33. $c2->push(1);
  34. });
  35. fibonacci($c1, $c2);