Javascript Promises:遍历所有对象键数组然后解析

新手上路,请多包涵

我有这个 JS 对象:

 let setOfWords = {
    "nouns": [
        "work",
        "construction",
        "industry"
    ],
    "verbs": [
        "work"
    ],
}

我正在使用调用 REST 资源的谷歌翻译 API,所以我需要等待每个翻译的响应,然后解析相同的对象结构,但使用翻译的单词。

 function translateByCategory(){
    let translatedObj = {};
    return new Promise(function(resolve, reject){

        Object.keys(obj).forEach(function(category){
            if (translatedObj[category] == undefined) {
                translatedObj[category] = [];
            }
            setOfWords.forEach(function(word){
                google.translate(word, 'es', 'en').then(function(translation){
                    translatedObj[category].push(translation.translatedText);
                });
            });

            // return the translatedObj until all of the categories are translated
            resolve(translatedObj);
        });
    });
}

原文由 Gus 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 548
2 个回答

您可以使用 Promise.all() 等待所有承诺履行(或第一次拒绝):

 var translateRequests = [];

Object.keys(setOfWords).forEach(function(category){
        setOfWords[category].forEach(function(word){
            translateRequests.push(google.translate(word, 'es', 'en'));
        });
    });
});

Promise.all(translateRequests).then(function(translateResults){
    //do something with all the results
});

请参阅文档: Promise.all()

原文由 hackerrdave 发布,翻译遵循 CC BY-SA 4.0 许可协议

个人承诺需要与 Promise.all() 聚合,从根本上说,这就是所缺少的。

但是您可以通过减少对 Google 服务的调用次数来做得更好。

谷歌翻译 API 允许通过传递一组单词而不是每次调用一个单词来一次翻译多个文本字符串,虽然可能不是价格优势,但可以为您提供性能优势 - 谷歌目前对其翻译服务“按字符”收费,不是“每次通话”。

我找不到 google.translate() 的任何文档--- 但是,根据一些假设,您可以编写:

 function translateByCategory(obj, sourceCode, targetCode) {
    let translatedObj = {};
    var promises = Object.keys(obj).map(function(key) {
        return google.translate(obj[key], sourceCode, targetCode).then(function(translations) {
            translatedObj[key] = translations.map(function(t) {
                return t.translatedText || '-';
            });
        }, function(error) {
            translatedObj[key] = [];
        });
    });
    return Promise.all(promises).then(function() {
        return translatedObj;
    });
}

如果这不起作用,那么 本文档 说明了如何直接调用谷歌的 RESTful 翻译服务。

你应该能够写:

 function translateTexts(baseParams, arrayOfStrings) {
    let queryString = baseParams.concat(arrayOfStrings.map(function(str) {
        return 'q=' + encodeURIComponent(str);
    })).join('&');

    return http.ajax({ // some arbitrary HTTP lib that GETs by default.
        url: 'https://translation.googleapis.com/language/translate/v2?' + queryString,
    }).then(function(response) {
        return response.data.translations.map(function(t) {
            return t.translatedText || '-';
        });
    }, function(error) {
        translatedObj[key] = []; // on error, default to empty array
    });
}

function translateByCategory(obj, sourceCode, targetCode) {
    let baseParams = [
        'key=' + MY_API_KEY, // from some outer scope
        'source=' + sourceCode, // eg 'en'
        'target=' + targetCode // eg 'es'
    ];
    let translatedObj = {};
    let promises = Object.keys(obj).map(function(key) {
        return translateTexts(baseParams, obj[key]).then(function(translations) {
            translatedObj[key] = translations;
        }, function(error) {
            translatedObj[key] = []; // on error, default to empty array
        });
    });
    return Promise.all(promises).then(function() {
        return translatedObj;
    });
}

在任何一种情况下,请按如下方式调用:

 let setOfWords = {
    "nouns": [
        "work",
        "construction",
        "industry"
    ],
    "verbs": [
        "work"
    ],
};

translateByCategory(setOfWords, 'en', 'es').then(function(setOfTranslatedWords) {
    console.log(setOfTranslatedWords);
});

原文由 Roamer-1888 发布,翻译遵循 CC BY-SA 3.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题