5

我以自己的方式履行承诺,但我坚持使用我的用例。我有一组转换器函数(每个函数都是一个承诺并修改一些 JSON 结构)。

让我展示一些代码。

假设这是我的 JSON 结构(数组)

var data = [{a: 1, b:2}, {a:3, b:4}]; 

transformFunction是以某种方式修改数据的变换函数的定义。这两个函数将cd属性添加到上述 JSON 结构中:

var transformFunctions = { // 

    transform1: function (data) {  // This function adds `c` property to each object from `a`
        return new Promise(function (resolve) {
             for (var i = 0; i < data.length; i++) {
                 data[i].c = data[i].a;
             }
             return resolve(data);
        })
    },

    transform2: function (data) {  // This function adds `d` property to each object from `c`
        return new Promise(function (resolve) {
             for (var i = 0; i < data.length; i++) {
                 data[i].d = data[i].c;
             }
             return resolve(data);
        })
    },
    ...
}

来自 UI 用户的 指定他应该使用哪些转换器功能以及以什么顺序使用。假设他选择了这样的正常顺序:

var userTransformList = ['transform1', 'transform2'];

方法应该修改数据,transform1结果应该传递给transform2方法。

我在看:Promise.all但它似乎不关心承诺的顺序,最重要的是它需要将先前的结果传递给下一个承诺。

4

2 回答 2

5

注意:正如 adeneo 在评论中指出的那样,只有在处理异步代码时才使用 Promise。

  1. 创建要执行的函数数组。并确保他们都返回一个 Promise。

  2. 然后,您可以Promise.reduce通过在每次迭代中返回执行当前承诺返回函数的结果来将初始值减少到转换后的最终值。

  3. 最后,您可以附加一个then处理程序以获取实际值和一个catch处理程序,以防万一承诺被拒绝。

假设我们有两个像这样的变换函数。

注:我再说一遍。你永远不应该将 Promises 与这些函数一起使用。仅当您处理的函数确实是异步的时,您才应该使用 Promise。

// Just add a property called `c` to all the objects and return a Promise object
function transform1(data) {
    return Promise.resolve(data.map(function(currentObject) {
        currentObject.c = currentObject.a + currentObject.b;
        return currentObject;
    }));
}

// Just add a property called `d` to all the objects and return a Promise object
function transform2(data) {
    return Promise.resolve(data.map(function(currentObject) {
        currentObject.d = currentObject.a + currentObject.b + currentObject.c;
        return currentObject;
    }));
}

然后你可以转换原始值,像这样

Promise.reduce([transform1, transform2], function (result, currentFunction) {
        return currentFunction(result);
    }, [{a: 1, b: 2}, {a: 3, b: 4}])      // Initial value
    .then(function (transformedData) {
        console.log(transformedData);
    })
    .catch(function (err) {
        console.error(err);
    });

输出

[ { a: 1, b: 2, c: 3, d: 6 }, { a: 3, b: 4, c: 7, d: 14 } ]
于 2015-06-09T14:15:22.310 回答
4

你可以像往常一样链接 Promises:使用.then().

假设您有以下两个转换:

function increment(x) {
    return Promise.resolve(x + 1);
}

function double(x) {
    return Promise.resolve(2 * x);
}

在实际场景中,这些将执行异步工作。你可以:

increment(1).then(double)

但是,您不知道转换的顺序或数量。让我们将它们放入一个数组中,并then()一一进行:

var transformations = [increment, double]

var promise = Promise.resolve(1);

for (var i = 0; i < transformations.length; i++)
  promise = promise.then(transformations[i]);

catch()您可以在开始之前、完成之后甚至每次转换时附加处理程序。

如果您要应用数百个转换,这效率不高。在这种情况下,您应该使用thefourtheye在他的回答中提出的建议reduce()

于 2015-06-09T14:39:11.747 回答