一个问题引发的思考

在我学习javascript的事件时,有一个小任务是使用JS来实现 li 列表项在鼠标悬浮时会有背景阴影的动态效果,很自然想到用for 来为每个列表项添加onmouseover 和 onmouseout事件来改变和恢复 li 的类名。

如下:

 <script type="text/javascript">
var Lis = document.getElementsByTagName("li"); function addevent() {
for (var i = 0; i < Lis.length; i++) {
Lis[i].onmouseover = function() {
console.log(i);
Lis[i].className = "lihover";
}
Lis[i].onmouseout = function() {
Lis[i].className = "";
}
}
}
addevent();
// console.log(i);
</script>

看起来很有道理的代码会什么不能正常工作?

先看一下为什么

在第8行和11行,通过改变 li 元素的classname来实现鼠标悬浮的动态效果改变,根据我以前学习的语言(C和JAVA)这明显是不对的,怎么能在函数内使用外面的局部变量呢,可是浏览器为什么没有报错,我在7行加了一句console.log(i);看一下

浏览器输出了3 并报错:closure.html:44 Uncaught TypeError: Cannot set property 'className' of undefined

当这两个事件发生时,函数会执行,函数会访问Lis 和 i 这两个变量,但是注意循环和事件函数不是同时执行的,事件函数发生时i的 已经是3了, 而Lis里并没有Lis[3]这个元素。

可是这两个函数为什么能访问外面的局部变量呢,答案是 闭包 (closure)

闭包初识

闭包就是在创建函数时为其保存一份创建时的外部环境,所以这两个事件函数能够访问到i这个变量,虽然访问的是循环执行完后的i的值。这也已经很神奇了对吧,源自函数式编程的魔法。那么怎么让其访问的 i 是函数自身被创建时的 i 呢,我再创建一个函数专门用来返回这个事件函数,如下:

   function makeevents(i) {
console.log(i);
return function() {
Lis[i].className = "lihover";
}
} function addevent() {
for (var i = 0; i < Lis.length; i++) {
Lis[i].onmouseover = makeevents(i);
Lis[i].onmouseout = function() {
this.className = "";
}
}
}
addevent();

我只是在mouseover事件用了这个机制,在mouseout使用了this关键词这个等最后在讨论。

在浏览器里运行一下发现效果已经实现了,并且在页面加载完后控制台就打印出了0 1 2

makeevent函数的作用正是创建闭包,在每一次循环创建一个,这样Lis[i]引用的就是正确的 li 元素了,看起来好像很繁琐的样子,事实上我们因为可以对makeevent传入参数来改变返回的函数的一些特点。这好像面向对象的工厂模式对吧。事实上我们完全可以用闭包来实现面向对象的封装。

关于this

我在第二个事件函数里使用了this,它表示了响应这个事件的当前对象,也即表示当前的列表项。并且在必要时还可以往Lis[i]元素对象里加入其他东西,如下:

      Lis[i].index = i;
Lis[i].hehe = "hehe";
Lis[i].onmouseout = function() {
console.log(this.hehe);
Lis[this.index].className = "";
}

这样写也是可以的。通过对象本身可以动态的添加属性这一个JavaScript的特点来完成的。

是的,JavaScript真是一门神奇的语言

初识JavaScript闭包的更多相关文章

  1. Javascript闭包和C#匿名函数对比分析

    C#中引入匿名函数,多少都是受到Javascript的闭包语法和面向函数编程语言的影响.人们发现,在表达式中直接编写函数代码是一种普遍存在的需求,这种语法将比那种必须在某个特定地方定义函数的方式灵活和 ...

  2. Day15 HTML补充、初识JavaScript

    一.上节回顾 上节回顾: HTML 头部信息:编码.title.style.link(导入css文件) 身体: 内联 块级 --->inline-block(既有内联效果又有块级效果) a标签: ...

  3. 《Web 前端面试指南》1、JavaScript 闭包深入浅出

    闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...

  4. JavaScript 闭包深入浅出

    闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...

  5. JavaScript闭包(Closure)

    JavaScript闭包(Closure) 本文收集了多本书里对JavaScript闭包(Closure)的解释,或许会对理解闭包有一定帮助. <你不知道的JavsScript> Java ...

  6. javascript闭包理解

    //闭包理解一 function superFun(){ var _super_a='a'; function subfuc(){ console.log(_super_a); } return su ...

  7. Javascript闭包深入解析及实现方法

    1.什么是闭包 闭包,官方对闭包的解释是:一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.闭包的特点:1. 作为一个函数变量的一个引用,当函数返回时 ...

  8. javascript闭包和作用域链

    最近在学习前端知识,看到javascript闭包这里总是云里雾里.于是翻阅了好多资料记录下来本人对闭包的理解. 首先,什么是闭包?看了各位大牛的定义和描述各式各样,我个人认为最容易一种说法: 外部函数 ...

  9. JavaScript闭包深入解析

    for (var i=1; i<=5; i++) { setTimeout( function timer() { console.log( i ); }, i*1000 ); } --上面这段 ...

随机推荐

  1. xlsx导入成--json

    这两天遇到大难题了,就是这个   xlsx   导入问题,之前用的xlsx.full.min.js,写的导入,结果不兼容ie浏览器,研究这个也好长时间,网上居然还没有搜到合适的,自己写从xlsx官网上 ...

  2. EOS 上线前,先搞懂这两个基本概念

    如果你曾经尝试在本地运行 EOS 测试节点,会发现编译.运行并不是特别复杂,但官方教程里两个概念很容易把人搞晕: Account(账户)和 Wallet (钱包). EOS 的 Wallet 跟其他区 ...

  3. 【重要】使用Git命令行上传到GitHub上

    [本人GitHub账号:] 用户名:chenhongshuang 密码:shuangshuang6300 邮箱:2452420371@qq.com 进入GitHub账号后 1·新建项目文件名称例dem ...

  4. [总结] 第二类Stirling数

    上一道例题 我们来介绍第二类Stirling数 定义 第二类Stirling数实际上是集合的一个拆分,表示将n个不同的元素拆分成m个集合的方案数,记为 或者 .和第一类Stirling数不同的是,集合 ...

  5. 学习java第一章

    本人是一名5年工作的人了,出来社会也比较早,工作经验比起刚刚出社会的大学生要和很多了,知道社会的现实与无奈,我为什么选择想学java昵,肯定受到了朋友的影响的,接下来就讲讲我学习java的过程. 1. ...

  6. 复习C#

    (1)public共有访问.该修饰符可用于类和结构的成员,可用于命名空间下直接定义的类型,对于类和结构成员,如果声明为共有的,那么除自身的成员,外部成员也可以访问 (2)private限制为私有访问. ...

  7. javaScript设计模式-创建型设计模式

    我们大家一听到设计模式就感觉设计模式是一个高端的东西,到底什么是设计模式呢?其实设计模式也就是我们的前辈在写代码的时候遇到的问题,提出的解决方案,为了方便人与人之间的交流,取了个名字,叫做设计模式. ...

  8. Java基础学习笔记十八 异常处理

    什么是异常?Java代码在运行时期发生的问题就是异常. 在Java中,把异常信息封装成了一个类.当出现了问题时,就会创建异常类对象并抛出异常相关的信息(如异常出现的位置.原因等). 异常的继承体系 在 ...

  9. beta冲刺1-咸鱼

    前言:这篇算是开始补之前的开端,毕竟beta阶段我们从前面开始就有在陆续做了. 今天的工作: 接收了新成员*1,然后几个人聚了一下,并且讨论了一下目前遇到的问题,以及目前需要处理的问题. 目前遇到的问 ...

  10. 项目Beta冲刺Day6

    项目进展 李明皇 今天解决的进度 进行前后端联动调试 明天安排 完善程序运行逻辑 林翔 今天解决的进度 服务器端发布消息,删除消息,检索消息,个人发布的action 明天安排 图片功能遇到问题,微信小 ...