async.auto中的任务结果

时间:2015-01-09 04:19:52

标签: javascript node.js asynchronous

我对async.auto中从一个任务到另一个任务的结果逻辑感到困惑。例如,在以下代码逻辑中,我向task1中的模型添加了一些数据,最初是initialtask的输出,finalTasktask1的模型中添加的数据被反映出来在results.initialTask1也是如此。同样,task2中添加的数据会反映在results.initialTask1中的finalTask

总结 results.initialTask1中的所有results.task1[0]results.task2[0]results.task3[0]finalTask都相同。这是async.auto的逻辑吗?或者它是否像C ++中的指针引用那样导致task1中模型的任何变化,它也反映在initialTask中的模型中?

async.auto({
    initialTask: function(callback) {
        //Do some operations
        callback(null, name, initialModels);
    },
    task1: ['initialTask', function(callback, results) {
        var models = results.initialTask[1];
        //Add some more data to models
        callback(null, models);
    }],
    task2: ['initialTask', function(callback, results) {
        var models = results.initialTask[1];
        //Add some more data to models
        callback(null, models);
    }],
    task3: ['initialTask', function(callback, results) {
        var models = results.initialTask[1];
        //Add some more data to models
        callback(null, models);
    }],
    finalTask: ['task1', 'task2', 'task3', function(callback, results) {
        //Here the followings are the same: results.initialTask[1], results.task1[0], results.task2[0], results.task3[0]                               
    }]
});

我正在寻找能帮助我确定逻辑与否的任何答案?我不一定要找任何官方文件或......

2 个答案:

答案 0 :(得分:6)

这是预期的行为。基本上async.auto将按照它认为必要的顺序执行所有功能。因此,在您的情况下,将首先调用initialTask。然后将并行调用task1task2task3。最后,将使用结果调用finalTask。所有值相同的原因与JavaScript call-by-sharing有关,这意味着如果您更改函数参数本身,则它不会影响输入参数的项目。如果您更改参数的内部,它将携带到项目。

更多信息here

示例:

async.auto({
// this function will just be passed a callback
readData: async.apply(fs.readFile, 'data.txt', 'utf-8'),
showData: ['readData', function(results, cb) {
    // results.readData is the file's contents
    // ...
}]
}, callback);

async.auto({
get_data: function(callback) {
    console.log('in get_data');
    // async code to get some data
    callback(null, 'data', 'converted to array');
},
make_folder: function(callback) {
    console.log('in make_folder');
    // async code to create a directory to store a file in
    // this is run at the same time as getting the data
    callback(null, 'folder');
},
write_file: ['get_data', 'make_folder', function(results, callback) {
    console.log('in write_file', JSON.stringify(results));
    // once there is some data and the directory exists,
    // write the data to a file in the directory
    callback(null, 'filename');
}],
email_link: ['write_file', function(results, callback) {
    console.log('in email_link', JSON.stringify(results));
    // once the file is written let's email a link to it...
    // results.write_file contains the filename returned by write_file.
    callback(null, {'file':results.write_file, 
'email':'user@example.com'});
}]
}, function(err, results) {
console.log('err = ', err);
console.log('results = ', results);
});

答案 1 :(得分:0)

async.auto是非常有用且功能强大的功能,由Async Lib提供。它有3个字段 1任务 2-并发 3-回调

在Async.auto中,除了第一个函数之外,每个函数都依赖于它的父函数,如果任何函数在执行期间会出现任何错误。那么它们的子函数或说...它们的下定义函数将不会被进一步执行,回调将发生错误,主回调将立即返回错误

1-任务: - 一个对象  2-并发: - 一个可选的整数,用于确定可以并行运行的最大任务数。默认情况下,尽可能多。 3-回调: - 返回响应

exapmle -

 AnyService.prototype.forgetPassword = function (res, email, isMobile, callback) {
    Logger.info("In AnyService service forgetPassword email...", email);
    db.User.findOne({
        email: email.toLowerCase(),
        deleted: false
    }, function (err, user) {
        if (!user) {
            configurationHolder.responseHandler(res, null, configurationHolder.LoginMessage.registerFirst, true, 403)
        } else {
            async.auto({
                token: function (next, results) {
                    return gereratePasswordToken(next, email, user, isMobile);
                },
                sendMail: ['token', function (next, result) {
                    return SendMailService.prototype.forgetPasswordMail(next, result.token, email, user.fullName);
                  }]
            }, function (err, result) {
                if (err == null && result != null) {
                    configurationHolder.ResponseUtil.responseHandler(res, null,      configurationHolder.LoginMessage.forgotPassword, false, 200)
                } else {
                    callback(new Error(configurationHolder.errorMessage.oops))
                }
            })
        }
    });
  }