4

我通过在订阅后添加项目修改了教程https://www.dartlang.org/docs/tutorials/streams/中的一些示例:

import 'dart:async';

main() {
  var data = new List<int>();
  var stream = new Stream.fromIterable(data);  // create the stream

  // subscribe to the streams events
  stream.listen((value) {       //
    print("Received: $value");  // onData handler
  });                           //
  data.add(1);
}

运行这个程序后,我得到了:

Uncaught Error: Concurrent modification during iteration: _GrowableList len:1.
Stack Trace: 
#0      ListIterator.moveNext (dart:_collection-dev/iterable.dart:315)
#1      _IterablePendingEvents.handleNext (dart:async/stream_impl.dart:532)
#2      _PendingEvents.schedule.<anonymous closure> (dart:async/stream_impl.dart:661)
#3      _asyncRunCallback (dart:async/schedule_microtask.dart:18)
#4      _createTimer.<anonymous closure> (dart:async-patch/timer_patch.dart:11)
#5      _Timer._createTimerHandler._handleTimeout (timer_impl.dart:151)
#6      _Timer._createTimerHandler.<anonymous closure> (timer_impl.dart:166)
#7      _RawReceivePortImpl._handleMessage (dart:isolate-patch/isolate_patch.dart:93)

data.add(1)在添加侦听器之前放置按预期工作。

我检查了有关 Stream 的文档并没有发现我做错了什么。我原以为听众会在最好的情况下被解雇,而在最坏的情况下不会被解雇,但也不例外。

这是预期的行为吗?如果是,请描述原因。

4

2 回答 2

4

异常来自您在迭代时尝试修改列表。这是 Dart (*) 中未指定的行为,使用的实现只是选择抛出异常。虽然它被 中发生的异步事情混淆了Stream.fromIterable,但它基本上与您尝试执行此操作相同:

var data = [1,2,3];
for(var d in data) {
    print(d);
    data.add(d+10);
}

如果您将您的包裹data.add在另一个异步调用中,例如使用Timer.run(() => data.add(2)),它会“工作”。我的意思是它不会抛出异常。

Received: 2仍然不会被打印。该流将仅发送当时已在列表中new Stream.fromIterable被调用的元素。之后,流被关闭(onDone将被调用),对原始列表的修改不会发送给您的侦听器。

(*) 来源:iterator.dart在 SDK 1.1.3 中——“如果迭代的对象在迭代过程中发生变化,则行为未指定。” 为什么 api.dartlang.org 上的文字不同,我无法理解。

编辑

要回答评论中的问题:一种方法是使用StreamController.

// or new StreamController<int>.broadcast(), if you want to listen to the stream more than once
StreamController s = new StreamController<int>();
// produce periodic errors
new Timer.periodic(new Duration(seconds: 5), (Timer t) {
    s.isClosed ? t.cancel() : s.addError("I AM ERROR");
});
// add some elements before subscribing
s.add(6);
s.add(9);
// this will close the stream eventually
new Timer(new Duration(seconds: 20), () => s.close());
// start listening to the stream
s.stream.listen((v) => print(v), 
        onError: (err) => print("An error occured: $err"), 
        onDone: () => print("The stream was closed"));
// add another element before the next event loop iteration
Timer.run(() => s.add(4711));
// periodically add an element
new Timer.periodic(new Duration(seconds: 3), (Timer t) {
    s.isClosed ? t.cancel() : s.add(0);
});
// one more (will be sent before 4711)
s.add(4);
于 2014-02-21T14:25:03.990 回答
1

迭代时无法修改列表。对于您的示例,您需要一个没有此限制的迭代(例如自定义实现)。

于 2014-02-21T14:06:58.607 回答