Fork me on GitHub

javascript中异步和闭包产生的困惑

这里我不打算大谈特谈什么是异步,什么是闭包,这些内容在博客园都已经写的够多的了,但是这些内容出现的多,并不代表所有初学者都已经撑握了,所以我还是打算,用一个比较常见的示例来分析一下,或许能让对这个问题有困惑的同学有一种顿悟的感觉。我在上一篇博客《从一道面试题分析闭包>中已经分析过什么是闭包了,但是那个例子应用的场景比较复杂,不适合初学者理解,这里我举一个更常见的例子.

        假如有这样一个需求:点击菜单中的每一项,显示所点击的内容,对应的内容页面如下:

<!DOCTYPE html>
<html>
<head>
<title>test</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
 
</head>
<body>
 <ul>
     <li>list 0</li>
     <li>list 1</li>
     <li>list 2</li>
     <li>list 3</li>
     <li>list 4</li>
 </ul> 
 <script type="text/javascript">

 </script>
</body>
</html>

我先来一段有问题的js代码,它是这样实现的:

  var items =  document.querySelectorAll('li');
  var len = items.length;

  for(var i =0;i<len;i++){
      items[i].onclick = function(){
          alert('list '+i)
      }
  }

这时候,我发现每一个li标签都邦定了点击事件,看起来运行的很好,可是当我多点几个,发现问题就来了,无论我点哪个li标签,弹出的都是list 5.这又是什么原因呢?

热心的网友马上给出回复说,因为弹出的i是循环之后的值。这样回答,那问题又来了:

1 :既然弹出的是循环之后的值,为什么每一个li标签上又都邦上了事件呢?

2 : 岂不是只有第5个li 元素可以邦上事件,而页面上根本不存这个元素,岂不是点击的时候,就该报错了呢?

说明这样回复,不但没有解释清楚问题,反而增加了更多的疑问,难道不是吗?这里产生疑问的根本原因在于有一个异步过程。循环代码是同步的,而点击操作是异步的。

我用一个游戏来演示这个过程:

操场上站着4个小朋友(对应4个 li元素),老师挨个的告诉他们游戏规则(对应循环),规则是这样的:呆会老师会举一个小黑板,然后点你名字的时候,你就告诉老师黑板上写的是什么字。(对应onclick所设定的function)。游戏开始了,老师在黑板上写了1,觉得不好,改为2,接着又改成3,最后改成4.小朋友们看着老师改来改去,等了好久才开始游戏。

可是无论点哪个小朋友,他回答的都是“4”。因为他们看到的都是老师最后写的那个数字。

现在我们再回过来看刚才的代码。

 var items =  document.querySelectorAll('li');
  var len = items.length;

  for(var i =0;i<len;i++){
     //当i=0的时候,即items[0]所对应的元素,这是确定的。
    //items[i]这里执行的是一个取值操作
      items[i].onclick = function(){
          //而这里边的i却要等到这个函数运行时才能确定是多少
         //函数什么时候运行,肯定发生在循环之后了。因为点击的速度显然是比不过cpu运算的速度的
          alert('list '+i)
      }
  }

刚才那个游戏,显然不是老师期望的。这次她把小黑板换成写好字的小纸片,挨个讲规则的时候顺便把纸片传给他们每一个人。这样每个小朋友手里都拿了一个属于自己的数字。老师点名字的时候,他们都报出了自己纸片上的数字。老师为自己的创意感到满意。

那我们这个程序,要怎么把i做成小纸片事先传给每一个li元素呢?

方法一:

在li上做一个标记,点的时候取这个标记上的值。

  var items =  document.querySelectorAll('li');
  var len = items.length;

  for(var i =0;i<len;i++){
      items[i].setAttribute('i',i);
      items[i].onclick = function(){
          i = this.getAttribute("i");
          alert('list '+i)
      }
  }

方法二:

  利用闭包的特性

  

  var items =  document.querySelectorAll('li');
  var len = items.length;

  for(var i =0;i<len;i++){
      items[i].onclick = (function(){
          var t = i;
          return function(){
              alert('list '+t)
          } 
      })()
  }

 

方法三:利用闭包比较难理解,我们换一个方式 

  var items =  document.querySelectorAll('li');
  var len = items.length;

  for(var i =0;i<len;i++){
       
      items[i].onclick = function(t){
              alert('list '+t)
      }.bind(this,i)
  }

总结一下:

以上虽然用了三种形式的小纸片进行传参,但是目的都是为了保证在循环之后,每个li的回调函数上的参数都能确定下来。这种小纸片在解决异步问题上,是一个有用的技巧。

关于异步的问题,还有很多,由于时间关系,就不再一一列举了,有兴趣的同学可以@我,一起学习。

如果您觉得这文章对您有帮助,请点击【推荐一下】,想跟我一起学习吗?那就【关注】我吧!

 

posted on 2014-11-02 14:48  bjtqti  阅读(1513)  评论(2编辑  收藏  举报