javascript - 强制异步调用以同步方式运行

标签 javascript asynchronous reactjs microservices

在我的 React 应用程序中,我尝试根据其他三个值来计算一个值。我已将所有计算逻辑包含到后端,这是我进行异步调用的微服务。我异步尝试获取计算值的函数位于许多同步 Hook 的中间。

在UI层,我调用想要返回最终结果的函数(异步返回)。被调用的函数调用另一个函数,该函数又调用另一个函数,该函数返回一个新的 Promise。请参阅下面的代码:

// DateUI.js (layer 1)
selectDate(dateField, flight, idx, saved, momentTime, e) {
    if (moment(momentTime).isValid()) {
        if (dateField == "StartDate") {
            // The initial problematic function call, need to set endDate before I continue on
            let endDate = PlanLineActions.calculateFlightEndDate(periodTypeId, numberOfPeriods, momentTimeUnix);

            flight.set("EndDate", endDate);
        }

        this.theNextSyncFunction(..., ..., ...);
    }
}


// DateActions.js (layer 2)
calculateFlightEndDate(periodTypeId, numberOfPeriods, startDate) {
    let plan = new Plan();

    plan.getFlightEndDate(periodTypeId, numberOfPeriods, startDate).then(function(response) {
        // response is JSON: {EndDate: "12/05/2016"}
        response.EndDate;
    }, function(error) {
        log.debug("There was an error calculating the End Date.");
    });
}


// DateClass.js (layer 3)
getFlightEndDate(periodTypeId, numberOfPeriods, startDate) {
    let path = '/path/to/microservice';
    return this.callServer(path, 'GET', {periodTypeId: periodTypeId, numberOfPeriods: numberOfPeriods, startDate: startDate});
}


// ServerLayer.js (layer 4)
callServer(path, method = "GET", query = {}, data, inject) {
    return new Promise((resolve, reject) => {
        super.callServer(uri.toString(),method,data,inject).then((data) => {
            resolve(data);
        }).catch((data) => {
            if (data.status === 401) {
                AppActions.doRefresh();
            }
            reject(data);
        });
    });
}

我的印象是,因为 ServerLayer.js(第 4 层)返回 new Promise (因此 DateClass.js(第 3 层)),调用 plan.getFlightEndDate(...).then(function(response) {...在响应返回“解决”或“拒绝”之前不会完成。目前尚未发生这种情况,因为 DateUI.js(第 1 层)中的代码将继续调用 this.theNextSyncFunction ,然后在约 50 毫秒后使用正确的数据进行解析。

如何在继续 selectDate() 之前强制 DateUI.js(第 1 层)中的 PlanLineActions.calculateFlightEndDate(...) 完成响应?

最佳答案

如果某些内容像 ajax 调用一样超出事件循环,则无法强制同步。您将需要如下所示的东西:

PlanLineActions.calculateFlightEndDate(periodTypeId, numberOfPeriods, momentTimeUnix)
  .then(endDate => {
    this.theNextSyncFunction(..., ..., ...);
  })

为了做到这一点,calculateFlightEndDate 还需要返回一个 Promise,因此 Promise 是可链接的,这是一件好事。

calculateFlightEndDate(periodTypeId, numberOfPeriods, startDate) {
  let plan = new Plan();

  // return promise!
  return plan.getFlightEndDate(periodTypeId, numberOfPeriods, startDate).then(response => {
    return response.EndDate; // must return here
  }, error => {
    log.debug("There was an error calculating the End Date.");
  });
}

这样就可以了..还有一件事:你在服务器调用中加倍了 promise 。如果某个东西有 .then 它已经是一个 promise ,所以你可以直接返回它。无需包装new Promise( promise 中的 promise ..不需要!)

callServer(path, method = "GET", query = {}, data, inject) {
  // just return!
  return super.callServer(uri.toString(),method,data,inject).then((data) => {
    return data;
  }).catch((data) => {
    if (data.status === 401) {
      AppActions.doRefresh();
    }
    throw data; // throw instead of reject
  });
}

关于javascript - 强制异步调用以同步方式运行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37445823/

相关文章:

javascript - 为什么我的表单水印脚本不再起作用?

C# 套接字异步与多线程

reactjs - React/Redux 服务器端渲染初始状态

javascript - 使用异步等待自动 try catch 路由器 Controller

java - 我如何获得 Vector<type> Arraylist<type> 作为 call() 的返回值?

css - 使用 flexbox 和 svg(使用 Sandbox) react 星级评分组件

javascript - Reactjs:在 setState 中使用时 Obj 被重新分配,有些奇怪

javascript - 根据选择输入在 Angular 中显示一个元素

javascript - 变量在 for 循环中变为未定义

javascript - Firefox扩展js对象初始化