异步/等待不等待

IT技术 node.js reactjs webpack babeljs ecmascript-next
2021-04-28 15:17:34

我遇到了一个我不完全理解的问题。我觉得可能有一些我没有掌握的概念,可以优化的代码,可能还有一个错误被抛出以进行很好的衡量。

为了大大简化整体流程:

  1. 向外部 API 发出请求
  2. 解析和扫描返回的 JSON 对象以获取链接引用
  3. 如果找到任何链接引用,则会发出其他请求以使用真实的 JSON 数据填充/替换链接引用
  4. 替换所有链接引用后,将返回原始请求并用于构建内容

这是原始请求(#1):

await Store.get(Constants.Contentful.ENTRY, Contentful[page.file])

Store.get 表示为:

async get(type, id) {
    return await this._get(type, id);
}

其中调用:

_get(type, id) {
    return new Promise(async (resolve, reject) => {
        var data = _json[id] = _json[id] || await this._api(type, id);

        console.log(data)

        if(isAsset(data)) {
            resolve(data);
        } else if(isEntry(data)) {
            await this._scan(data);

            resolve(data);
        } else {
            const error = 'Response is not entry/asset.';

            console.log(error);

            reject(error);
        }
    });
}

API调用是:

_api(type, id) {
    return new Promise((resolve, reject) => {
        Request('http://cdn.contentful.com/spaces/' + Constants.Contentful.SPACE + '/' + (!type || type === Constants.Contentful.ENTRY ? 'entries' : 'assets') + '/' + id + '?access_token=' + Constants.Contentful.PRODUCTION_TOKEN, (error, response, data) => {
            if(error) {
                console.log(error);

                reject(error);
            } else {
                data = JSON.parse(data);

                if(data.sys.type === Constants.Contentful.ERROR) {
                    console.log(data);

                    reject(data);
                } else {
                    resolve(data);
                }
            }
        });
    });
}

当一个条目被返回时,它被扫描:

_scan(data) {
    return new Promise((resolve, reject) => {
        if(data && data.fields) {
            const keys = Object.keys(data.fields);

            keys.forEach(async (key, i) => {
                var val = data.fields[key];

                if(isLink(val)) {
                    var child = await this._get(val.sys.linkType.toUpperCase(), val.sys.id);

                    this._inject(data.fields, key, undefined, child);
                } else if(isLinkArray(val)) {
                    var children = await* val.map(async (link) => await this._get(link.sys.linkType.toUpperCase(), link.sys.id));

                    children.forEach((child, index) => {
                        this._inject(data.fields, key, index, child);
                    });
                } else {
                    await new Promise((resolve) => setTimeout(resolve, 0));
                }

                if(i === keys.length - 1) {
                    resolve();
                }
            });
        } else {
            const error = 'Required data is unavailable.';

            console.log(error);

            reject(error);
        }
    });
}

如果找到链接引用,则会发出其他请求,然后将生成的 JSON 注入到原始 JSON 中以代替引用:

_inject(fields, key, index, data) {
    if(isNaN(index)) {
        fields[key] = data;
    } else {
        fields[key][index] = data;
    }
}

请注意,我正在使用async, await, 和Promise's 我相信他们的预期庄园。最终发生的事情:对引用数据的调用(获得 _scan 的结果)最终发生在原始请求返回之后。这最终会为内容模板提供不完整的数据。

关于我的构建设置的其他信息:

  • npm@2.14.2
  • 节点@4.0.0
  • webpack@1.12.2
  • 巴别@5.8.34
  • babel-loader@5.4.0
1个回答

我相信问题出在您的forEach来电中_scan作为参考,请参阅使用 ES7 驯服异步野兽中的这段话

然而,如果你尝试使用异步函数,那么你会得到一个更微妙的错误:

let docs = [{}, {}, {}];

// WARNING: this won't work
docs.forEach(async function (doc, i) {
  await db.post(doc);
  console.log(i);
});
console.log('main loop done');

这将编译,但问题是这将打印出:

main loop done
0
1
2

发生的事情是主函数提前退出,因为await实际上在子函数中。此外,这将同时执行每个Promise,这不是我们想要的。

教训是:当你的异步函数中有任何函数时要小心。await只会暂停其父功能,因此检查它在做什么,你真的觉得它在做什么。

所以每次forEach调用的迭代都是并发运行的;他们不是一次执行一个。一旦符合条件的函数i === keys.length - 1完成,promise 就会被解析并_scan返回,即使调用 via 的其他异步函数forEach仍在执行。

您需要将forEacha更改为 amap以返回一组Promise,然后您可以await*从中_scan(如果您想同时执行它们,然后在它们全部完成后调用某些东西),或者执行它们-time 如果您希望它们按顺序执行。


附带说明一下,如果我没看错的话,您的一些异步函数可以稍微简化一下;记住这一点,而await荷兰国际集团的async函数调用返回一个值,只需调用它返回另一个Promise,并返回从valueasync功能是一样的返回一个Promise解析为在非该值async的功能。因此,例如,_get可以是:

async _get(type, id) {
  var data = _json[id] = _json[id] || await this._api(type, id);

  console.log(data)

  if (isAsset(data)) {
    return data;
  } else if (isEntry(data)) {
    await this._scan(data);
    return data;
  } else {
    const error = 'Response is not entry/asset.';
    console.log(error);
    throw error;
  }
}

同样,_scan可以是(假设您希望forEach主体同时执行):

async _scan(data) {
  if (data && data.fields) {
    const keys = Object.keys(data.fields);

    const promises = keys.map(async (key, i) => {
      var val = data.fields[key];

      if (isLink(val)) {
        var child = await this._get(val.sys.linkType.toUpperCase(), val.sys.id);

        this._inject(data.fields, key, undefined, child);
      } else if (isLinkArray(val)) {
        var children = await* val.map(async (link) => await this._get(link.sys.linkType.toUpperCase(), link.sys.id));

        children.forEach((child, index) => {
          this._inject(data.fields, key, index, child);
        });
      } else {
        await new Promise((resolve) => setTimeout(resolve, 0));
      }
    });

    await* promises;
  } else {
    const error = 'Required data is unavailable.';
    console.log(error);
    throw error;
  }
}