多次解决承诺是否安全?

2022-08-30 03:00:37

我的应用程序中有一个 i18n 服务,其中包含以下代码:

var i18nService = function() {
  this.ensureLocaleIsLoaded = function() {
    if( !this.existingPromise ) {
      this.existingPromise = $q.defer();

      var deferred = this.existingPromise;
      var userLanguage = $( "body" ).data( "language" );
      this.userLanguage = userLanguage;

      console.log( "Loading locale '" + userLanguage + "' from server..." );
      $http( { method:"get", url:"/i18n/" + userLanguage, cache:true } ).success( function( translations ) {
        $rootScope.i18n = translations;
        deferred.resolve( $rootScope.i18n );
      } );
    }

    if( $rootScope.i18n ) {
      this.existingPromise.resolve( $rootScope.i18n );
    }

    return this.existingPromise.promise;
  };

这个想法是用户会打电话并等待承诺得到解决。但是,鉴于该函数的目的只是确保加载了区域设置,因此用户多次调用它完全没问题。ensureLocaleIsLoaded

我目前只存储一个 promise,如果用户在从服务器成功检索区域设置后再次调用该函数,则解析它。

据我所知,这按预期工作,但我想知道这是否是一种正确的方法。


答案 1

正如我目前所理解的承诺,这应该是100%好的。唯一需要理解的是,一旦解析(或拒绝),那就是对于延迟对象 - 它就完成了。

如果您再次调用其承诺,您将立即获得(第一个)已解决/拒绝的结果。then(...)

对 的其他调用将不会产生任何影响。resolve()

下面是一个涵盖这些用例的可执行代码段:

var p = new Promise((resolve, reject) => {
  resolve(1);
  reject(2);
  resolve(3);
});

p.then(x => console.log('resolved to ' + x))
 .catch(x => console.log('never called ' + x));

p.then(x => console.log('one more ' + x));
p.then(x => console.log('two more ' + x));
p.then(x => console.log('three more ' + x));

答案 2

我刚才也遇到过同样的事情,确实一个承诺只能解决一次,另一次尝试将无济于事(没有错误,没有警告,没有调用)。then

我决定像这样解决它:

getUsers(users => showThem(users));

getUsers(callback){
    callback(getCachedUsers())
    api.getUsers().then(users => callback(users))
}

只需将您的函数作为回调传递,并根据需要多次调用它!希望这是有道理的。