如何在Dart中创建一个空白的Future+如何返回当前正在进行的Future?

ct2axkht  于 2023-04-27  发布在  其他
关注(0)|答案(6)|浏览(167)

我试图创建一个服务器端的Dart类,它执行各种与数据相关的任务。所有这些任务都依赖于数据库的初始化。问题是数据库的初始化是异步发生的(返回一个Future)。我首先尝试将init代码放入构造函数中,但已经放弃了这种方法,因为它似乎不可行。
我现在试图弄清楚如何在任何访问数据的方法调用中强制DB初始化作为第一步。换句话说,当下面调用attemptLogin()时,我想首先检查DB是否已经初始化,并在必要时初始化它。
然而,有两个障碍。如果数据库还没有初始化,代码很简单-初始化数据库,然后使用返回的future的then()方法来执行剩下的函数。如果数据库还没有初始化,我应该把then()方法附加到什么?
第二个相关的问题是,当数据库正在初始化,但这个过程还没有完成时,会发生什么?我如何拉入并返回这个“正在进行中”的Future?
这是我试图讨论的代码的基本要点:

class DataManager {
  bool DbIsReady = false;
  bool InitializingDb = false;
  Db _db;

  Future InitMongoDB() {
    print("Initializing MongoDB");
    InitializingDb = true;
    _db = new Db("mongodb://127.0.0.1/test");
    return _db.open().then((_) {
      DbIsReady = true;
      InitializingDb = false;
    });
  }  

  Future<List> attemptLogin(String username, String password) {
    Future firstStep;
    if ((!DbIsReady) && (!InitializingDb) {
       Future firstStep = InitMongoDB()
    }
    else if (InitializingDb) {
       // Need to return the InitMongoDB() Future that's currently running, but how?
    }
    else {
       // How do I create a blank firstStep here? 
    }

    return firstStep.then((_) {
      users = _db.collection("users");
      return // ... rest of code cut out for clarity
    });
  }
}

先谢谢你的帮助,
格雷格

wvyml7n5

wvyml7n51#

刚回来

return new Future<bool>.value(true); 
// or any other value instead of `true` you want to return.
// or none
// return new Future.value();
gajydyqb

gajydyqb2#

让未来继续活着:

class DataManager {
  Future _initializedDb;

  Future initMongoDb() { ... }

  Future<List> attemptLogin(String username, String password) {
    if (_initializedDb == null) {
       _initializedDb = initMongoDB();
    }
    return _initializedDb.then((db) {
      users = db.collection("users");
      return // ... rest of code cut out for clarity
    });
  }
}

您可能需要注意错误情况。如果您想处理initMongoDB中或之后的错误,则由您决定。

jyztefdp

jyztefdp3#

可能的解决方案之一:

import "dart:async";

void main() {
  var dm = new DataManager();
  var selectOne = dm.execute("SELECT 1");
  var selectUsers = dm.execute("SELECT * FROM users");
  var users = selectOne.then((result) {
    print(result);
    return selectUsers.then((result) {
      print(result);
    });
  });

  users.then((result) {
    print("Goodbye");
  });
}

class Event {
  List<Function> _actions = new List<Function>();
  bool _raised = false;

  void add(Function action) {
    if (_raised) {
      action();
    } else {
      _actions.add(action);
    }
  }

  void raise() {
    _raised = true;
    _notify();
  }

  void _notify() {
    if (_actions.isEmpty) {
      return;
    }

    var actions = _actions.toList();
    _actions.clear();
    for (var action in actions) {
      action();
    }
  }
}

class DataManager {
  static const int _STATE_NOT_INITIALIZED = 1;
  static const int _STATE_INITIALIZING = 2;
  static const int _STATE_READY = 3;

  Event _initEvent = new Event();
  int _state = _STATE_NOT_INITIALIZED;

  Future _init() {
    if (_state == _STATE_NOT_INITIALIZED) {
      _state = _STATE_INITIALIZING;
      print("Initializing...");
      return new Future(() {
        print("Initialized");
        _state = _STATE_READY;
        _initEvent.raise();
      });
    } else if (_state == _STATE_INITIALIZING) {
      print("Waiting until initialized");
      var completer = new Completer();
      _initEvent.add(() => completer.complete());
      return completer.future;
    }

    return new Future.value();
  }

  Future execute(String query, [Map arguments]) {
    return _init().then((result) {
      return _execute(query, arguments);
    });
  }

  Future _execute(String query, Map arguments) {
    return new Future.value("query: $query");
  }
}

输出:

Initializing...
Waiting until initialized
Initialized
query: SELECT 1
query: SELECT * FROM users
Goodbye

我认为存在更好的解决方案,但这只是试图回答你的问题(如果我正确理解你)。

P.S.编辑于2014年7月11日

稍微修改了一下(带错误处理)示例。

import "dart:async";

void main() {
  var dm = new DataManager();
  var selectOne = dm.execute("SELECT 1");
  var selectUsers = dm.execute("SELECT * FROM users");
  var users = selectOne.then((result) {
    print(result);
    return selectUsers.then((result) {
      print(result);
    });
  });

  users.then((result) {
    print("Goodbye");
  });
}

class DataManager {
  static const int _STATE_NOT_INITIALIZED = 1;
  static const int _STATE_INITIALIZING = 2;
  static const int _STATE_READY = 3;
  static const int _STATE_FAILURE = 4;

  Completer _initEvent = new Completer();
  int _state = _STATE_NOT_INITIALIZED;

  Future _ensureInitialized() {
    switch (_state) {
      case _STATE_NOT_INITIALIZED:
        _state = _STATE_INITIALIZING;
        print("Initializing...");
        new Future(() {
          print("Initialized");
          _state = _STATE_READY;
          // throw null;
          _initEvent.complete();
        }).catchError((e, s) {
          print("Failure");
          _initEvent.completeError(e, s);
        });

        break;    
      case _STATE_INITIALIZING:
        print("Waiting until initialized");
        break;
      case _STATE_FAILURE:
        print("Failure detected");
        break;
      default:
        print("Aleady intialized");
        break;
    }

    return _initEvent.future;
  }

  Future execute(String query, [Map arguments]) {
    return _ensureInitialized().then((result) {
      return _execute(query, arguments);
    });
  }

  Future _execute(String query, Map arguments) {
    return new Future.value("query: $query");
  }
}
dsekswqp

dsekswqp4#

对于那些仍然想知道如何在Dart中创建一个空白的Future并在以后完成它们的人,你应该像下面的例子一样使用Completer类。

class AsyncOperation {
  final Completer _completer = new Completer();

  Future<T> doOperation() {
    _startOperation();
    return _completer.future; // Send future object back to client.   
  }

  // Something calls this when the value is ready.   
  void finishOperation(T result) {
    _completer.complete(result);   
  }

  // If something goes wrong, call this.   
  void _errorHappened(error) {
      _completer.completeError(error);   
  }
}
j91ykkif

j91ykkif5#

Future在Dart中是不可空的,这意味着你必须将它初始化为一个值。如果你不这样做,Dart会抛出以下错误:
错误:应初始化字段,因为其类型“Future”不允许为null。
要初始化Future,请参见以下示例:

Future<String> myFutureString = Future(() => "Future String");

这里的“Future String”是一个String,因此上面的代码返回Future的一个示例。
因此,在谈到如何创建一个空白/空的Future的问题时,我使用了以下代码来初始化一个空的Future列表。

Future<List> myFutureList = Future(() => []);

我发现这个链接对于理解Flutter和Dart中的Futures非常有用:https://meysam-mahfouzi.medium.com/understanding-future-in-dart-3c3eea5a22fb

k2arahey

k2arahey6#

要创建void类型的黑色Future,例如:

onRefresh: () async {},

这里,一个类型为Future<void>的空函数被分配给onRefresh属性。

相关问题