如何在 Javascript 中使用循环生成事件处理程序?

IT技术 javascript
2021-01-11 16:57:11

例如,我有 10 个从 AJAX 响应生成的标签:

<a href="#" id="b1">b1</a>
<a href="#" id="b2">b2</a>
<a href="#" id="b3">b3</a>
<a href="#" id="b4">b4</a>
<a href="#" id="b5">b5</a>
<a href="#" id="b6">b6</a>
<a href="#" id="b7">b7</a>
<a href="#" id="b8">b8</a>
<a href="#" id="b9">b9</a>
<a href="#" id="b10">b10</a>

我需要通过循环为每个人分配 onclick 事件:

for(i=1; i<11; i++) {
    document.getElementById("b"+i).onclick=function() {
        alert(i);
    }
}

这不起作用,它只将 onclick 分配给最后一个标签并警告“11”。我怎样才能让它工作?我不想使用 jQuery。

3个回答

您的所有处理程序都共享相同的i变量。

您需要将每个处理程序放入一个单独的函数中,该函数将其i作为参数,以便每个处理程序都有自己的变量:

function handleElement(i) {
    document.getElementById("b"+i).onclick=function() {
        alert(i);
    };
}

for(i=1; i<11; i++) 
    handleElement(i);
test.html:11 未捕获的类型错误:无法设置 null 的属性“onclick”
2021-04-09 16:57:11

闭包就是你要找的:

for(i=1; i<11; i++) {
    (function(i) {
        document.getElementById("b"+i).onclick=function() {
            alert(i);
        };
    })(i);
}
我会说 Caballero 有一个他们不想要的闭包,他们想要一个闭包破坏者来强制立即评估i. 这就是您的自执行功能正在做的事情。
2021-03-18 16:57:11
比拥有单独的功能要优雅得多。
2021-03-26 16:57:11
关闭/不关闭...我不在乎!这个匿名/自调用解决方案很漂亮:) 我会勾选这个。
2021-04-01 16:57:11

有两种方法可以在这个问题上使用闭包。这个想法是为事件处理程序使用的每次迭代创建一个带有“i”变量快照的范围。

解决方案#1(正如凯文所提到的):

for(i=1; i<11; i++) {
    (function(num) {

       document.getElementById("b"+num).addEventListener('click', function() {
            alert(num);
       });

    })(i);
}

解决方案#2:

for (i=1; i<11; i++) {
    document.getElementById("b"+i).addEventListener('click', (function(){
        var num = i;
        return function() {
            alert(num);
        }
    })());
}