0

我不是一个真正的承诺忍者,我知道我做错了什么。但是我找不到我所遇到的一些特殊/相似的问题。

问题:我为 IndexedDB 使用 Dexie.js 包装器,它是异步的。我有一个全局数据库,它指向其他一些 dexie 数据库。

function handleDatabases() {
    var result = [];

    db.jobs.orderBy('title').filter(function(job) {
        return job.someBooleanCondition;
    }).each(function(job, cursor) {
        let jobDetails = new Dexie(job.correspondingDB);
        jobDetails.version(1).stores({
            details: 'key,value1,value2'
        }); 
        jobDetails.details.get(someKey).then(function(detail) {
            result.push({job: job, detail: detail});
        })
    }).catch(function(error) {
        console.log(error);
    });
    handleResult(result);
}

我已经用一种可能很奇怪的形式为 SO 重写了它,但最终目标是我可以使用数组result来处理一些更新。但是,由于它是异步的,因此它始终为空,直到您在控制台中检查它,它永远不会为空。如何将其重写为同步?

4

1 回答 1

4

当结果仅异步可用时,您不能期望返回结果。

所以你必须一直坚持承诺(每次都返回),并且让你的函数也返回一个承诺。调用者必须使用then(或await如果支持)允许(异步)访问结果。

与其将 推入{job: job, detail: detail}结果变量,不如将其返回。它将成为 的承诺价值jobDetails.details.get(..).then(..)。如果您也返回它,您将获得一系列承诺,然后可以通过以下方式解决Promise.all

避免创建新的 Promises,因为这通常会导致Promise 构造函数 antipattern

还要避免使用在多个回调中使用而不作为参数传递的变量(如results )。而是尝试构造并返回该数组作为承诺值,以便可以在下一个then回调中使用它。

这是建议的(未经测试的)代码:

function handleDatabases() {
    db.jobs
    .orderBy('title')
    .filter(job => job.someBooleanCondition)
    .toArray(jobs =>
        jobs.map(job => {
            let jobDetails = new Dexie(job.correspondingDB);
            jobDetails.version(1).stores({
                details: 'key,value1,value2'
            });
            return jobDetails.details.get(someKey)
                   .then(detail => ({job: job, detail: detail}))
        }) // is returned
    )
    .then(result => Promise.all(result))
    .then(handleResult)
    .catch(error => console.log(error));
}
于 2017-04-02T00:03:15.500 回答