在Dart中创建一个循环流

时间:2015-07-25 12:01:37

标签: stream dart

我尝试编写聊天应用程序与计算机聊天。用户可以编写消息并获得计算机的响应。聊天记录可能如下所示:

user: Hi
computer: Hello
user: What's your name?
computer: Bot
...

我的基于循环流的设计灵感来自Cycle.js的想法。我有一个用户消息流,它被转换为计算机消息流,而这些消息又是用户消息流的输入:

    |----> user message stream ---->|
    |                               |
transform                       transform
    |                               |
    |<-- computer message stream <--|

此代码已经有效:

import 'dart:io';
import 'dart:async';

void main() {
  cycle(computer, user);
}

typedef Stream<T> Transform<T>(Stream<T> input);

void cycle(Transform aToB, Transform bToA) {
  var aProxy = new StreamController.broadcast();
  var b = aToB(aProxy.stream);
  var a = bToA(b);
  aProxy.add('start'); // start with user
  aProxy.addStream(a);
}

Stream<String> user(Stream<String> computerMessages) {
  computerMessages = computerMessages.asBroadcastStream();
  computerMessages.listen((message) => print('computer: $message'));
  return computerMessages.map((message) {
    stdout.write('user: ');
    return stdin.readLineSync();
  });
}

Stream<String> computer(Stream<String> userMessages) {
  var messages = <String, String>{
    "Hi": "Hello",
    "What's your name?": "Bot"
  };
  return userMessages.map((m) => messages.containsKey(m) ? messages[m] : 'What?');
}

只有一个问题。您需要一个起始值才能运行循环流。因此,我将此行放在我的函数cycle

aProxy.add('start'); // start with user

实际上,这个逻辑属于我的函数user,因为cycle不应该知道初始值。而且,我不喜欢打印初始值。它应该只触发用户输入流。因此,我更改了cycleuser

void cycle(Transform aToB, Transform bToA) {
  var aProxy = new StreamController.broadcast();
  var b = aToB(aProxy.stream);
  var a = bToA(b);
  aProxy.addStream(a);
}

Stream<String> user(Stream<String> computerMessages) {
  computerMessages = computerMessages.asBroadcastStream();
  computerMessages.listen((message) => print('computer: $message'));
  var requestInput = new StreamController<String>.broadcast();
  requestInput.add('start'); // start with user
  requestInput.addStream(computerMessages); // continue on computer response
  return requestInput.stream.map((message) {
    stdout.write('user: ');
    return stdin.readLineSync();
  });
}

但是通过此更改,我的应用程序立即终止,而stdout中没有消息。怎么了?

1 个答案:

答案 0 :(得分:0)

我找到了解决方案。在StreamController中创建普通而不是广播user

Stream<String> user(Stream<String> computerMessages) {
  computerMessages = computerMessages.asBroadcastStream();
  computerMessages.listen((message) => print('computer: $message'));
  var requestInput = new StreamController<String>();
  requestInput.add('start'); // start with user
  requestInput.addStream(computerMessages); // continue on computer response
  return requestInput.stream.map((message) {
    stdout.write('user: ');
    return stdin.readLineSync();
  });
}

即使我找到了解决方案,但遗憾的是我无法解释广播StreamController的问题。 null已成功添加requestInput.add(null);,但奇怪的是,requestInput.addStream(computerMessages);未完成,computerMessages的事件也未添加到requestInput。因此,requestInput关闭,应用程序终止。如果有人能提供进一步的解释,我将不胜感激。