0
点赞
收藏
分享

微信扫一扫

JavaScript之ES6规范中let的巧用(经典案例讲解)

微笑沉默 2022-04-26 阅读 43

html部分:

	<ul id="test">
        <li></li>
        <li></li>
        <li></li>
        <li></li>
    </ul>
    <ul>
        <li>dyklk</li>
    </ul>

要求:再点击某个li标签的时候,弹框输出其对应的顺序号(注意:本文js代码部分全为原生写法)。
错误写法:

var oLi =document.getElementById('test').getElementsByTagName('li');
   for(var i = 0; i < oLi.length; i++) {            
        oLi[i].onclick =function(event) {
            console.log(i+1);
        }
    }

此写法下:点击每个id为test的ul下的li,浏览器控制台输出的都是5。
改进办法①:

var oLi = document.getElementById('test').getElementsByTagName('li');
   for (var i = 0; i < oLi.length; i++) {
       oLi[i].index=i;
       oLi[i].onclick = function (event) {
           console.log(this.index+1);
       }
   }

改进办法②:使用let

var oLi =document.getElementById('test').getElementsByTagName('li');
  for(let i = 0; i < oLi.length; i++) {            
      oLi[i].onclick =function(event) {
          console.log(i+1);
      }
  }

解释:
错误写法:
for 循环中var关键字声明的变量i不具有块作用域,是一个全局变量,在页面打开时创建(变量声明提升),在页面关闭时销毁,当你点li标签的时候,页面已经加载完成,相关元素的事件绑定操作也已经完成,此时全局变量i经过for循环后已经变为4了,而当执行到输出语句时发现函数中没有i变量,根据作用域链往上寻找变量i,找到全局变量i后就输出了,结果当然不是你想要的啦。
改进办法①:在每次for循环中将索引 i 保存为对应li元素对象的一个属性,这样写直接让每个li元素对象与其顺序号绑定在一起了,所以肯定能成功啦。
改进办法②:循环执行oLi.length次,每次循环 i 的值不一样,,可以理解为每次循环都在一个新的块中进行,因为 i 是由let关键字声明的具有块级作用域的变量,不会影响作用域链,而onclick事件绑定的函数中没有 i 值,就会往上找到let声明的变量 i 来执行输出。(我自己是这么想的)

举报

相关推荐

0 条评论