1200字范文,内容丰富有趣,写作的好帮手!
1200字范文 > JavaScript循环绑定事件问题及解决方法

JavaScript循环绑定事件问题及解决方法

时间:2022-09-23 11:03:23

相关推荐

JavaScript循环绑定事件问题及解决方法

html代码

<ul><li>Lorem.<button>删除</button></li><li>Commodi.<button>删除</button></li><li>Tenetur!<button>删除</button></li><li>Corrupti?<button>删除</button></li><li>Obcaecati.<button>删除</button></li><li>Praesentium.<button>删除</button></li><li>Vero?<button>删除</button></li><li>Voluptates.<button>删除</button></li><li>Perferendis?<button>删除</button></li><li>Aliquam!<button>删除</button></li></ul>

如果我们想实现点击按钮删除一行li,第一反应是用一个for循环给按钮绑上点击事件,下面我们用这个思路写出来,看问题出在哪里

问题代码

var btns = document.querySelectorAll('button');for(var i = 0;i < btns.length;i++){btns[i].onclick = function(){console.log(i);// 每次都打印 10btns[i].parentElement.remove();}}

运行点击删除,却出现报错

不管点击哪个按钮,输出的都是10。 以我的理解是因为for循环是瞬发的,事件是绑定上了,但是这个函数是点击时调用的,绑定事件的时候i并没有被传入执行函数里,执行函数里没有声明i这个变量,所以它会去外部寻找i,此时循环早已结束,外部的i的值等于10;

解决方案

方案1

var btns = document.querySelectorAll('button');for(var i = 0;i < btns.length;i++){btns[i].onclick = function(){console.log(i);// 每次都打印 10this.parentElement.remove();}}

这个方法是利用this关键字,涉及到了this指向的问题,这里是btns这个对象调用这个函数,所以此时的this是指向这个对象本身。

方案2

var ul = document.getElementsByTagName('ul')[0];ul.onclick = function(e){// 判断当前点击的目标元素是不是button,只有点击的元素时button 才能做删除操作if(e.target.nodeName === 'BUTTON'){// 一个元素的父节点一定是一个元素节点e.target.parentElement.remove();}}

用事件委托实现这个功能,button不再处理事件,委托给父元素,利用事件对象Event进行判断,

判断当前点击的目标元素是不是button,只有点击的元素时button 才能做删除操作;

方案3

var btns = document.querySelectorAll('button');for (var i = 0; i < btns.length; i++) {(function fn(index) {btns[i].onclick = function() {btns[i].parentElement.remove();}})(i)}

利用闭包解决,将一个函数的内部的函数返回到函数外部去,这一现象可以称作闭包;此时btns为全局变量,每次循环会生成一个立即执行函数,立即执行函数里的事件函数给到了一个全局变量btns,所以也是闭包。

每个立即执行函数里都会传入i;这个i的值保存在这个立即执行函数的作用域里,所以此时点击按钮时,查询i这个值会在这个立即执行函数的作用域里找,并不会去找到全局的i

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。