1

我正在循环一些元素,然后添加新元素,这些元素在单击时应该操纵这些元素。这很难解释,所以请看一下这个 Fiddle 让它更清楚:

http://jsfiddle.net/pgFcn/3/

有趣的部分是这段代码(为简洁起见,伪代码):

for (var i = 0; i < divs.length; i++) {
    var div = divs[i];

    someOtherElement.addEventListener("click", function () {
        testDiv(div); // always refers to the last div because variable is overwritten next loop
    });
}

我希望 testDiv 调用分别引用 div 1、div 2、div 3,但相反,它们都引用 div 3,因为该变量在下一次循环迭代中被覆盖。我该如何解决这个问题?

4

2 回答 2

5

这是一个经典的问题。通常是这样解决的:

for (var i = 0; i < divs.length; i++) {
    (function(div){
      someOtherElement.addEventListener("click", function () {
        testDiv(div);
      });
    })(divs[i]);
}

要理解问题和解决方案,您必须知道在 JavaScript 中非全局变量的范围是声明它的函数的调用。这意味着

  • 在您的代码中,只有一个div变量
  • 在解决方案中,调用中间函数会产生不同的div变量
于 2013-07-09T12:34:17.363 回答
2

这是解决问题的另一种方法...

function testDiv(d) {
    // this doesn't work as expected, it always shows "Div 3"
    alert(d.innerText);
}

var divs = document.getElementsByTagName("div");

for (var i = 0; i < divs.length; i++) {
    var div = divs[i];
    var a = document.createElement("a");
    a.index = i;  
    a.innerText = "[this should point to Div" + (i+1) + "]";
    a.href = "#";
    a.addEventListener("click", function (e) {
        var e = e || window.event;
        var tg = e.target || e.srcElement;
        testDiv(divs[tg.index]);
    });
    document.body.appendChild(a);
}
于 2013-07-09T12:36:27.740 回答