初识JavaScript闭包
一个问题引发的思考
在我学习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闭包的更多相关文章
- Javascript闭包和C#匿名函数对比分析
C#中引入匿名函数,多少都是受到Javascript的闭包语法和面向函数编程语言的影响.人们发现,在表达式中直接编写函数代码是一种普遍存在的需求,这种语法将比那种必须在某个特定地方定义函数的方式灵活和 ...
- Day15 HTML补充、初识JavaScript
一.上节回顾 上节回顾: HTML 头部信息:编码.title.style.link(导入css文件) 身体: 内联 块级 --->inline-block(既有内联效果又有块级效果) a标签: ...
- 《Web 前端面试指南》1、JavaScript 闭包深入浅出
闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...
- JavaScript 闭包深入浅出
闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...
- JavaScript闭包(Closure)
JavaScript闭包(Closure) 本文收集了多本书里对JavaScript闭包(Closure)的解释,或许会对理解闭包有一定帮助. <你不知道的JavsScript> Java ...
- javascript闭包理解
//闭包理解一 function superFun(){ var _super_a='a'; function subfuc(){ console.log(_super_a); } return su ...
- Javascript闭包深入解析及实现方法
1.什么是闭包 闭包,官方对闭包的解释是:一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.闭包的特点:1. 作为一个函数变量的一个引用,当函数返回时 ...
- javascript闭包和作用域链
最近在学习前端知识,看到javascript闭包这里总是云里雾里.于是翻阅了好多资料记录下来本人对闭包的理解. 首先,什么是闭包?看了各位大牛的定义和描述各式各样,我个人认为最容易一种说法: 外部函数 ...
- JavaScript闭包深入解析
for (var i=1; i<=5; i++) { setTimeout( function timer() { console.log( i ); }, i*1000 ); } --上面这段 ...
随机推荐
- Path通过Selenium模拟浏览器抓取,Windows 64解决selenium.common.exceptions.WebDriverException: Message: 'geckodriver' executable needs to be in PATH.方法
1.下载geckodriver.exe: 下载地址:https://github.com/mozilla/geckodriver/releases请根据系统版本选择下载:(如Windows 64位系统 ...
- VMware安装Centos6.4及CentOS的基本设置
1.vmware安装vmware tools实现与本地磁盘文件夹的共享 2.设置Centos网络 vmware采用nat方式.虚拟机网络使用虚拟网卡(VMware Network Adapter VM ...
- 给定n,求1/x + 1/y = 1/n (x<=y)的解数~hdu-1299~(分解素因子详解)
链接:https://www.nowcoder.com/acm/contest/90/F来源:牛客网 题目描述 给定n,求1/x + 1/y = 1/n (x<=y)的解数.(x.y.n均为正整 ...
- 使用IDEA搭建Spring Boot入门项目
简介 Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程.该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置 ...
- Tram
Tram 题目大意:给你一个图,这个图上有n点和边.点上有开关,开关开始指向一条道路,拨动开关可以使开关指向由开关出发的任意一条路径,读入,a,b,求,至少要拨动几次才能从a点走到b点. 注释:n&l ...
- linux小白成长之路1————通过Parallels安装CentOS虚拟机
以下是通过Mac版Parallels安装CentOS虚拟机的教程: 1.在Parallels向导中选择"下载CentOS",点击"继续":  2.点击&quo ...
- 关于使用Mybatis的使用说明(一)【未完善待更新】
(一)搭建Mybatis环境 (1)先导入常用的jar包:并且需要将lib文件夹下的包导入到项目中 (2)创建config文件夹,配置log4j.properties文件 # Global loggi ...
- 仿京东树形菜单插件hovertree
hovertree是一个仿京东的树形菜单jquery插件,暂时有银色和绿色两种. 官方网址:http://keleyi.com/jq/hovertree/欢迎下载使用 查看绿色效果:http://ke ...
- Java基础学习笔记四 Java基础语法
数组 数组的需求 现在需要统计某公司员工的工资情况,例如计算平均工资.最高工资等.假设该公司有50名员工,用前面所学的知识完成,那么程序首先需要声明50个变量来分别记住每位员工的工资,这样做会显得很麻 ...
- java基础之类与对象
[类 & 对象] 1.类:具有一系列相同属性(特征)和方法(行为)的个体的集合,称为类. 类是一个抽象的概念,只能说类具有哪些属性,而不能直接对属性进行赋值. 例如:人类有身高的属性,但是不能 ...