dart - RxDart shareReplay 不“热”?
问题描述
我在我的应用程序中成功使用shareReplay
from RxDart
(0.24.1) 时遇到问题。我有一个Stream
并且我想“缓存”最新发出的值,以便任何“迟到”的侦听器立即获取它(流可以空闲更长的时间)。我没有做到这一点,所以我开始在我的应用程序之外进行实验,并且必须承认我并不完全理解正在发生的事情。
这是一些代码:
import 'package:rxdart/rxdart.dart';
void main() async {
final s = Stream.fromIterable([1, 2, 3]).shareReplay(maxSize: 10);
print('First:');
await for (final i in s) {
print(i);
}
print('Second:');
await for (final i in s) {
print(i);
}
}
我希望代码能够打印
First
1
2
3
Second
1
2
3
但它永远不会到达第二个await for
,它甚至不打印字符串Second:
;但是,程序成功完成,所以第一个await for
确实完成了(因为原始Stream
程序是有限的)。到底是怎么回事?
实际上,Stream
我所拥有的可能是无穷无尽的,因为事件之间的时间很长),而且我只对最后一个事件感兴趣,所以这里有一些模拟这个的代码:
import 'package:rxdart/rxdart.dart';
void main() async {
final s = Stream.periodic(Duration(seconds: 1), (e) => e).shareReplay(maxSize: 1);
// print(s.isBroadcast);
await Future.delayed(Duration(seconds: 3));
print('First');
await for (final i in s.take(1)) {
print(i);
}
print('Second');
await for (final i in s.take(1)) {
print(i);
}
}
为了确保Stream
罐头完成(也await for
可以完成),我使用take(1)
. 我预计输出是:
First
2
Second
2
(也许两个是三个?)我期望发生的是:
- 周期性流每秒开始发射(它是广播/热的)。
- 由于
Future.delayed
. - 第一个迟到的听众来了,得到了最新的值,2(或者可能是 3?)并且由于
take(1)
. - 第二个迟到的侦听器也来了并获得了最新的值,因为在此期间流没有发出任何其他东西(即使它发出了,该值也会大 1,这没关系)并且由于
take(1)
. - 该应用程序完成。
但是,输出是:
First
0
Second
Unhandled exception:
type '_TakeStream<int>' is not a subtype of type 'Future<bool>'
#0 _StreamIterator._onData (dart:async/stream_impl.dart:1067:19)
#1 _RootZone.runUnaryGuarded (dart:async/zone.dart:1384:10)
#2 _BufferingStreamSubscription._sendData (dart:async/stream_impl.dart:357:11)
#3 _BufferingStreamSubscription._add (dart:async/stream_impl.dart:285:7)
#4 _ForwardingStreamSubscription._add (dart:async/stream_pipe.dart:127:11)
#5 _TakeStream._handleData (dart:async/stream_pipe.dart:318:12)
#6 _ForwardingStreamSubscription._handleData (dart:async/stream_pipe.dart:157:13)
#7 _RootZone.runUnaryGuarded (dart:async/zone.dart:1384:10)
#8 _BufferingStreamSubscription._sendData (dart:async/stream_impl.dart:357:11)
#9 _BufferingStreamSubscription._add (dart:async/stream_impl.dart:285:7)
#10 _SyncBroadcastStreamController._sendData (dart:async/broadcast_stream_controller.dart:385:25)
#11 _BroadcastStreamController.add (dart:async/broadcast_stream_controller.dart:250:5)
#12 _StartWithStreamSink._safeAddFirstEvent (package:rxdart/src/transformers/start_with.dart:56:12)
#13 _StartWithStreamSink.onListen (package:rxdart/src/transformers/start_with.dart:37:11)
#14 forwardStream.<anonymous closure>.<anonymous closure> (package:rxdart/src/utils/forwarding_stream.dart:31:37)
#15 forwardStream.runCatching (package:rxdart/src/utils/forwarding_stream.dart:24:12)
#16 forwardStream.<anonymous closure> (package:rxdart/src/utils/forwarding_stream.dart:31:16)
#17 _runGuarded (dart:async/stream_controller.dart:847:24)
#18 _BroadcastStreamController._subscribe (dart:async/broadcast_stream_controller.dart:213:7)
#19 _ControllerStream._createSubscription (dart:async/stream_controller.dart:860:19)
#20 _StreamImpl.listen (dart:async/stream_impl.dart:493:9)
#21 DeferStream.listen (package:rxdart/src/streams/defer.dart:37:18)
#22 StreamView.listen (dart:async/stream.dart:1871:20)
#23 new _ForwardingStreamSubscription (dart:async/stream_pipe.dart:118:10)
#24 new _StateStreamSubscription (dart:async/stream_pipe.dart:341:9)
#25 _TakeStream._createSubscription (dart:async/stream_pipe.dart:310:16)
#26 _ForwardingStream.listen (dart:async/stream_pipe.dart:83:12)
#27 _StreamIterator._initializeOrDone (dart:async/stream_impl.dart:1041:30)
#28 _StreamIterator.moveNext (dart:async/stream_impl.dart:1028:12)
#29 main (package:shopping_list/main.dart)
<asynchronous suspension>
#30 _startIsolate.<anonymous closure> (dart:isolate-patch/isolate_patch.dart:301:19)
#31 _RawReceivePortImpl._handleMessage (dart:isolate-patch/isolate_patch.dart:168:12)
第一个await for
真正take(1)
使它完成的工作正如我所期望的那样,除了它获取初始值,这意味着流在侦听器到来之前没有发出任何状态(不是“热”)。第二个失败但有异常。我错过了什么?
编辑:我的理解shareReplay
可能一直是错误的,我虽然很热,因为https://pub.dev/documentation/rxdart/latest/rx/ReplaySubject-class.html说它很热。
我稍微更改了代码:
import 'package:rxdart/rxdart.dart';
void main() async {
final s = BehaviorSubject();
s.addStream(Stream.periodic(Duration(seconds: 1), (e) => e));
// print(s.isBroadcast);
await Future.delayed(Duration(seconds: 3));
print('First');
await for (final i in s.take(1)) {
print(i);
}
print('Second');
await for (final i in s.take(1)) {
print(i);
}
}
现在输出是:
First
2
Second
2
但是程序永远不会完成......
没有take(1)
输出永远不会到达第二个监听器,所以它确实有一些效果。我不明白。
编辑2:我想我可能知道为什么第一个示例shareReplay
不起作用,这是文档的摘录:
它会在第一次收听时自动开始发射项目,并在没有收听者时关闭。在我的情况下,第一个侦听器/
await for
订阅,流开始,当循环完成时,最后一个(也是唯一的)侦听器完成,所以从返回的流shareReplay
关闭。不过,仍然没有解释为什么该行print('Second');
不执行。
解决方案
您的最终程序未终止的原因是您的广播流仍在每秒发出事件。没有人在听,但它是一个广播流,所以它不在乎。它有一个每秒钟都在滴答作响的计时器,该计时器使程序保持活力,即使没有人会再听一次。您也可以通过这样做来保持程序的活力Timer.periodic(Duration(seconds: 1), (_) {});
。只要计时器处于活动状态,就可能发生某些事情。
不过,这次崩溃很有趣。它表明StreamIterator
该类的内部逻辑存在问题。我会调查的。
不能说任何保证有用的东西shareReplay
,我不熟悉 RxDart。我的猜测是第一个程序遇到问题是因为replayShared
流没有正确终止——也就是说,done
当它完成时没有发送事件。这可以解释程序提前终止的原因。第一个await for
是等待下一个事件,所以它不会结束并进入下一个循环,但是没有任何类型的实时计时器或预定事件,所以隔离只是关闭。快速阅读BeheaviorSubject
表明它像流控制器一样工作,因此您可能需要在addStream
完成后关闭它。也许:
s.addStream(...).whenComplete(s.close);
推荐阅读
- python - 如何在sqlalchemy模型中获取panrent类中子类的数据?
- javascript - 如何使用颜色选择器动态更改单击元素的颜色
- flutter - 颤动按下后退按钮时如何终止所有正在进行的操作?
- google-apps-script - 如何修复脚本以便我可以将唯一范围复制到另一个选项卡并避免复制列的标题?
- three.js - 如何仅将定向光应用于 Three.js 中的有限区域?
- c# - 如何在 DynamoDB AWS C# SDK 中使用 QueryAsync 和 ScanAsync?
- c# - EF Core 更新外键不会更新导航属性
- r - R - 根据变量删除重复行,但忽略 NA 和一些特定值
- bash - 命令的结果未分配给变量
- matlab - matlab如何测试测量值之间的显着差异?