2

我试图创建一个对象数组,然后访问数组中的对象属性,但它返回未定义。我调用 createObjArray() 函数,并在我执行 console.log(objArray[1]); 之后立即调用 它打印出具有所有属性的对象就好了。但是,如果我尝试执行 console.log(objArray[1].name); 萤火虫打印“未定义”。此外,在 firebug 中单步执行我的代码时,我可以将鼠标悬停在 objArray[1].name 上,它会显示正确的名称。这里发生的事情,让我抓狂。

var objArray = [];

function createObjectArray(numOfObjs) {

    for(var i=0; i<numOfObjs; i++) {

packages.push(initObj(i)); 

    }
 }

function initObj(i){
    var newPackage;
    var p = {};
    $.getJSON('.../package' + i + '.json', function(data) {
        newPackage = new Package(data);
        p.name = newPackage.name;
        p.id = i;      
    });
    return p;
 }
4

2 回答 2

1

这将起作用:

var objArray = [];

function createObjectArray(numOfObjs, callback) {
    var filledPackage = [];
    var nbLeft = numOfObjs;
    for(var i=0; i<numOfObjs; i++) {
        initObj(i, function(p){
            filledPackage.push(p);
            nbLeft--;
            if (nbLeft === 0){
                callback(filledPackage);
            }
        }); 
    }
 }

function initObj(i, callback){
    var newPackage;
    var p = {};
    $.getJSON('.../package' + i + '.json', function(data) {
        newPackage = new Package(data);
        p.name = newPackage.name;
        p.id = i;     
        callback(p);
    });
}

//Get a filled object array:
createObjectArray(5, function(filledArray){
    objArray = filledArray;
    //Code here will be executed AFTER all the $.getJSON queries have returned.
    //objArray is not empty.
});
//Code here will be executed WHILE the getJSON queries are running and
//while objArray is still empty. Due to the way the JS event loop works,
//it is impossible that code placed here will be able to use the content
//of objArray unless you call an async function such as anything AJAX or
//setTimeout, but that's iffy. Code you want to be executed once objArray
//has been filled should be inside of the callback above.

问题是 $.getJSON 是异步的,这意味着它不会自动返回结果。相反,你给它一个回调。回调是在收到结果后执行的函数。在这种情况下,回调是调用 $.getJSON 时创建的匿名函数。该回调从服务器接收结果,将其添加到数组中,然后检查数组是否已填充。由于 $.getJSON 函数我们正在执行异步代码,因此我们也必须异步返回结果。为此,我们要求 initObj 函数接收一个在完成后调用的函数(另一个回调)。我们调用该回调并将参数传递给它。然后我们再次通过回调返回填充的数组。

于 2012-11-07T00:33:06.733 回答
0

您对 $.getJSON 的调用是异步的。当 initObj() 返回 p 它仍然是一个空对象。

但是 initObj() 创建了一个闭包,该闭包捕获对 p 的引用,因此当 $.getJSON 返回时 p 被填充。

这就是为什么在填充数组后立即运行的代码中对象看起来为空的原因。但是,当您运行控制台命令时,异步调用已返回并且对象已被填充。

在继续处理阵列之前,您需要等待所有异步调用返回。一种方法是在您进行调用时增加一个计数器,并在调用返回时减少它,然后当最终调用返回时,计数器将降至零并继续处理。

或者,您可以设置一个 setTimout 循环,以便在填充所有项目时继续轮询数组检查。

如果您认为其中一个调用可能失败,这两种方法都是有风险的,但这种方法本身具有根本风险,因为您要进行多个 ajax 调用,因此您必须处理多个可能的故障。一次抓取所有数据会更干净,这样您就可以在 jQuery.ajax 的成功/错误处理程序中处理成功/错误状态一次。

于 2012-11-06T23:31:54.103 回答