如何使用 JS async/await 等待 AJAX 响应
我第一次在 Javascript 中使用 async/await 函数。我无法让脚本等待 AJAX 响应,然后再继续执行,然后读取/使用该响应。
我知道这里已经有很多关于 async/await 函数未按预期等待的问题,但其他问题的答案似乎都不适合我。
基本上,我想做的是循环遍历图层名称数组(用于
OpenLayers
地图),并且
forEach
图层名称我发送一个 AJAX 调用来从 MySQL 数据库中检索记录(如果存在)。然后我只需显示结果,转到下一个图层名称,发送下一个 AJAX 调用等。
这是我的代码:
async function getCellLayers() {
layerNames = [];
map.getLayers().forEach(function(layer) {
if (layer.get('type') == "cell") {
if (layer.getZIndex() == 100) {
layerNames.push(layer.get('name'));
if (layerNames.length == 1) {
fullExtent = layer.getSource().getExtent();
} else {
ol.extent.extend(fullExtent, layer.getSource().getExtent());
}
}
}
});
return layerNames;
}
async function getRecord(cell_date) {
$.ajax({
url: 'rec/getRecord/'+cell_date,
type: 'get',
dataType: 'json',
success: await function(response){
console.log("getRecord response: "+JSON.stringify(response));
return response['data'];
}
});
}
async function testAsyncAwaitFunction() {
let layerNames = await getCellLayers();
layerNames.forEach(async function(layerName) {
cell_date = layerName.substring(3)+"_"+window['currentImage'].substring(17,25);
console.log(cell_date+":");
let cellRecord = await getRecord(cell_date);
console.log("Matches: "+cellRecord.length);
console.log("testAsyncAwaitFunction response: "+JSON.stringify(cellRecord));
});
}
我期望在控制台中看到类似这样的内容:
cell101_20190202:
getRecord response: {"data": [{"id":1,"record":"cell101_20190202","value":"0.8"}]}
Matches: 1
testAsyncAwaitFunction response: {"data": [{"id":1,"record":"cell101_20190202","value":"0.8"}]}
cell102_20190202:
getRecord response: {"data": [{"id":2,"record":"cell102_20190202","value":"0.7"}]}
Matches: 1
testAsyncAwaitFunction response: {"data": [{"id":2,"record":"cell102_20190202","value":"0.7"}]}
[ ... and so on ... ]
但是我得到的却是:
cell101_20190202:
cell102_20190202:
(...)
getRecord response: {"data": [{"id":1,"record":"cell101_20190202","value":"0.8"}]}
getRecord response: {"data": [{"id":2,"record":"cell102_20190202","value":"0.7"}]}
(...)
getRecord response: {"data": [{"id":14,"record":"cell202_20190202","value":"0.6"}]}
(200x) Uncaught (in promise) TypeError: Cannot read property 'length' of undefined
getRecord response: {"data": [{"id":15,"record":"cell204_20190202","value":"0.5"}]}
(...)
我从未看到以
testAsyncAwaitFunction response
为前缀的
JSON.stringify
行,大概是因为在该 console.log 命令之前的行(尝试获取 cellRecord 的长度)由于 AJAX 响应尚未到达而失败。
我怀疑下面这行将是这里的关键:
let cellRecord = await getRecord(cell_date);
但是我不明白为什么那一行似乎没有“等待”尽管上面几行代码似乎运行良好:
let layerNames = await getCellLayers();
如果能得到更了解如何使用 async/await 的人的帮助,我将不胜感激。我更习惯使用 PHP 和 Python,很难将我的思维模式转变为异步思考。
这里有两件事:
- 您的
getRecord
函数不返回
Promise
,因此 await 不会等待任何内容
-
forEach
无法与异步函数一起使用,因为实现不会等待。
对于第一个问题,您可以通过执行以下操作来解决:
async function getRecord(cell_date) {
return $.ajax({
url: 'rec/getRecord/'+cell_date,
type: 'get',
dataType: 'json',
})
.then(response => response.data);
}
对于第二个问题,您可以通过这种方式运行循环来解决:
async function testAsyncAwaitFunction() {
let layerNames = await getCellLayers();
for (layerName of layerNames) {
cell_date = layerName.substring(3)+"_"+window['currentImage'].substring(17,25);
console.log(cell_date+":");
let cellRecord = await getRecord(cell_date);
console.log("Matches: "+cellRecord.length);
console.log("testAsyncAwaitFunction response: "+JSON.stringify(cellRecord));
}
}
但这样做会使所有内容逐一运行。您可以通过发送请求然后等待所有请求完成来做得更好,方法是使用
Promise.all
这样做:
const promises = []
for (layerName of layerNames) {
cell_date = layerName.substring(3)+"_"+window['currentImage'].substring(17,25);
console.log(cell_date+":");
promises.push(getRecord(cell_date));
}
const records = await Promise.all(promises)
将
getRecord
更改为此
function getRecord(cell_date) {
return $.ajax({
url: 'rec/getRecord/'+cell_date,
type: 'get',
dataType: 'json'
}).then(function(response){
console.log("getRecord response: "+JSON.stringify(response));
return response['data'];
});
}
并从代码中的所有位置删除
async
和
await
关键字,但以下两个部分的
testAsyncAwaitFunction
除外:
async function testAsyncAwaitFunction()
和
let cellRecord = await getRecord(cell_date);
否则您不需要它们。
它以前不会起作用,因为您的函数需要返回包含数据的承诺。您应该阅读
JavaScript 承诺
。 Async/Await 在很大程度上是这些的语法糖,用于处理异步代码。您拥有的唯一实际异步代码是对
getRecord
的调用。
关于异步的要记住的事情是,与异步使用的任何功能都有望返回承诺。 GetRecord应该返回您拥有的东西。 另外,虽然您的外部函数testasyncawaitfunction是异步的,并且您的foreach回调为异步,但您没有任何等待解决方案的所有承诺。
您想要此模式:
< << < <<代码> 974462135