async.Future async.Completer - 如果出错,如何“继续”

时间:2013-05-08 10:54:38

标签: dart

以下一些帮助将不胜感激。我正在编写一些控制台测试程序,我希望能够从终端输入一些参数(我不想使用命令行参数 - 参数太多)。我尝试了一些变化,但我找不到如何实现这一点。以下是我的终端输入测试的最新版本。这个程序的问题是如果遇到错误,完成程序会自动关闭,我想从Main()或fGetNumber()函数继续。虽然我可以看到为什么这个程序不起作用,但它说明了我需要实现的目标 - 重新输入数字,但我找不到如何实现这一目标。如果输入有效数字,则没有问题。如果输入的号码无效,我无法找到如何重新输入号码。

代码如下,我所遇到的问题以“//////////”:

突出显示
import "dart:async" as async;
import "dart:io";

void main() {
  fGetNumber("Enter Nr of Iterations : ", 0, 999999)
  .then((int iIters){   
    print ("In Main : Iterations selected = ${iIters}");
    if (iIters == null) {
      print ("In Main: Invalid Number of iterations : ${iIters}.");
    } else {
      fProcessData(iIters);
    }
    print ("Main Completed");       
  });
 }

async.Future<int> fGetNumber(String sPrompt, int iMin, int iMax) {
  print ("In fGetNumber");
  int iIters  = 0; 
  async.Completer<int> oCompleter = new async.Completer();

  while (!oCompleter.isCompleted) { ///////////   This loop does not work ///////  
    return fGetUserInput(sPrompt).then((String sIters) {
      iIters = int.parse(sIters);
      if (iIters < iMin || iIters > iMax) throw new Exception("Invalid");        
      oCompleter.complete(iIters);
      return oCompleter.future;
    }).catchError((_) => print ("Invalid - number must be from ${iMin} to ${iMax}")
    ).whenComplete(() => print ("fGetNumber - whenComplete"));// always gets here
  }
  print ("In fGetNumber (at end of function)");     //// it never gets here
}

async.Future<String>  fGetUserInput(String sPrompt) {
  print ("In fGetUserInput");
  async.Completer<String> oCompleter = new async.Completer(); 

  stdout.write(sPrompt);
  async.Stream<String> oStream = stdin.transform(new StringDecoder());
  async.StreamSubscription oSub;
  oSub = oStream.listen((String sData) {
    oCompleter.complete("$sData");
    oSub.cancel();
  });

  return oCompleter.future;
}

void fProcessData(int iIters) {
  print ("In fProcessData");
  for (int iPos = 1; iPos <= iIters; iPos++ ) {
    if (iPos%100 == 0) print ("Processed = ${iPos}");
  }
  print ("In fProcessData - completed ${iIters}");
} 

1 个答案:

答案 0 :(得分:3)

  

//此循环不起作用

当然它确实 - 你只输入一次,然后立即返回,因此离开循环和方法。

  

//总是到这里

这是因为whenComplete()总是在成功或出错时被调用。

  

//它永远不会到达

因为你已退出该方法。

那可以做些什么?

最简单的方法是不依赖fGetUserInput()。收听stdin中的fGetNumber,只有在输入有效时才能完成订阅/取消订阅:

async.Future<int> fGetNumber(String sPrompt, int iMin, int iMax) {

  print ("In fGetNumber");
  async.Completer<String> oCompleter = new async.Completer(); 

  stdout.write(sPrompt);
  async.Stream<String> oStream = stdin.transform(new StringDecoder());
  async.StreamSubscription oSub;
  oSub = oStream.listen((String sData) {
    try {
      int iIters = int.parse(sData);
      if (iIters < iMin || iIters > iMax) throw new Exception("Invalid"); 
      oCompleter.complete(iIters);
      oSub.cancel();
    } catch(e) {
      print("Invalid - number must be from ${iMin} to ${iMax}");
      stdout.write(sPrompt);
    }
  });

  return oCompleter.future;
}

有替代方案吗?

当然。有很多很多方法可以做到这一点。例如:

async.Future<int> fGetNumber(String sPrompt, int iMin, int iMax) {
  print ("In fGetNumber");
  async.Completer<int> oCompleter = new async.Completer();
  fGetUserInput(sPrompt, oCompleter, (String sIters) {
    try {
      int iIters = int.parse(sIters);
      if (iIters < iMin || iIters > iMax) throw new Exception("Invalid");
      return iIters;
    } catch(e) {
      print ("Invalid - number must be from ${iMin} to ${iMax}");
      stdout.write(sPrompt);
    }
    return null;
  });
  return oCompleter.future;
}

void fGetUserInput(String sPrompt, async.Completer oCompleter, dynamic inputValidator(String sData)) {
  print ("In fGetUserInput");
  stdout.write(sPrompt);
  async.Stream<String> oStream = stdin.transform(new StringDecoder());
  async.StreamSubscription oSub;
  oSub = oStream.listen((String sData) {
    var d = inputValidator(sData);
    if(d != null) {
      oCompleter.complete(d);
      oSub.cancel();
    }
  });
}

如果你真的觉得Dart团队应该解决一些问题,你可以写一个feature request。但Completer只能完成一次。无论你编写什么代码,你都不能循环再次完成它。