javascript - 不要在顺序链接的 Promise 中包含 q Promise 结果

标签 javascript promise q

我有以下顺序链接的 Q Promise,它填充了用户->属性->租户->提醒的层次结构:

    return getAllUsers()    // get all users of the system
    .then(function(users) {
        return Q.all(users.map(function(user) {
            return getUserProperties(user)  // foreach user of the system get their properties
            .then(function(properties) {
                user.properties = properties;
                return Q.all(properties.map(function(property) {
                    return getPropertyTenants(property) // foreach property get active tenants
                    .then(function(tenants) {
                        property.tenants = tenants;
                        return Q.all(tenants.map(function(tenant) {
                            return getTenantReminders(property, tenant) // foreach tenant get their payment reminders
                            .then(function(reminders) {
                                // if reminders.length == 0 we don't want to include this user
                                tenant.reminders = reminders;
                            });
                        }));
                    });
                }));
            }).thenResolve(user);
        }));
    });

此 promise 链的要点是获取租户的所有付款提醒,然后向每个用户发送一封电子邮件,其中包含租户提醒列表(每个用户)。如果 getTenantReminders() 没有解决任何提醒,那么理想情况下,顶级 getAllUsers() 中的用户不会包含在结果中。

我考虑过从 getTenantReminders().then() 处理程序中抛出一个特殊错误,该错误会沿着链向上并在 thenResolve(user) 中处理,然后不会返回用户。不确定这是否是正确的方法。

感谢任何建议。

最佳答案

正如塞斯所说,在这里保持理智的关键是避免厄运金字塔。这是我认为应该有效的方法(尽管显然我还没有测试过):

// Produces a function that takes an array of values as input,
// Applies func() to each of them, and merges the results into a 
// promise for a single array.
// func() is assumed to take a single value as input and 
// return an array or promise for an array
// (X -> Promise<Y[]>) -> (X[] -> Promise<Y[]>)
function applyAllAndMerge(func) {
    return function (values) {
        return Q.all(values.map(func))
        .then(function (arrays) {
            return Array.prototype.concat.apply([], arrays);
        });
    };
}

// returns a promise for an array of all tenant reminders for the
// specified property
// property -> Promise<tenantReminder[]>
function getTenantRemindersForProperty(property) {
    return getPropertyTenants(property)
    .then(applyAllAndMerge(function (tenant) {
        return getTenantReminders(property, tenant);
    }));
}

// returns a promise for an array of all tenant reminders for the 
// specified user
// user -> Promise<tenantReminder[]>
function getTenantRemindersForUser(user) {
    return getUserProperties(user)
    .then(applyAllAndMerge(getTenantRemindersForProperty));
}

// returns a promise for an object containing the specified
// user and all of their tenant reminders
// user -> Promise<{user, tenantReminder}[]>
function getUserAndTenants(user) {
    return getTenantRemindersForUser(user)
    .then(function (reminders) {
        return { user: user, reminders: reminders };
    });
}        

return getAllUsers()    // get all users of the system
.then(function(users) {
    return Q.all(users.map(getUserAndTenants));
})
.then(function (userReminders) {
    // get rid of users with no reminders
    var usersWithReminders = userReminders.filter(function (ur) {
        return ur.reminders.length > 0;
    });

    // use usersWithReminders
});

关于javascript - 不要在顺序链接的 Promise 中包含 q Promise 结果,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28905800/

相关文章:

javascript - 一次又一次重新加载img onReady函数

javascript - 使用 For 循环从字符串中删除最后一个字符(如果它是 "!") - JavaScript

javascript - Promise js方法时如何避免逻辑重复?

node.js - 如何使用带有 promise 的异步等待来获取结果?

javascript - "Non-constructor value passed to NewPromiseCapability"是什么意思?

javascript - 使用自定义指令后,angularjs ng-model 不起作用

javascript - 仅用于数字和连字符的正则表达式

javascript - XMLHttpRequest Promise,嵌套在一些同步代码中

node.js - 链式顺序任务

javascript - 如何在 mongoDB 中使用 $limit 和 $sort 与 Q.nbind()