Javascript 笔记与总结(1-5)闭包
【例1】
<script>
function t1(){
var age = 20;
function t2(){
alert(age);
}
return t2;
}
var tmp = t1();
var age = 99;
tmp();
</script>
弹出 20
【分析】
function t1(){
var age = ;
function t2(){
alert(age);
}
return t2;
}
在大部分的语言中,t1 被调用执行,则申请内存,并把其局部变量 push 入栈,t1 函数执行完毕,内部的局部变量,随着函数的退出而销毁,导致 age = 20 的局部变量消失。
因此,t1(); 执行完毕之后,按 C 语言的理解,t1 内的局部变量都释放了。
function t1(){
var age = ;
function t2(){
alert(age);
}
return t2;
}
在 js 中,t1 执行过程中,又生成了 t2,
而从作用域上来说,t2 能访问到 age = 20,
于是 “ age = 20 ” 没有消失,而是被 t2 捕捉到了,
同时 “ age = 20 ” 与返回的 t1 函数形打了个包返回来了,成了一个 “ 环境包 ”,
这个包属于 t2,所以叫 “ 闭包 ”。
t2 把周围的环境打了包,也就是说 t2 是由自己的生态环境的,
即 即使t1 执行完毕,通过 t2 ,依然能访问该变量:这种情况,函数并非孤立的函数,甚至把其周围的变量环境,形成了一个封闭的环境包,共同返回。
一句话概括:函数的作用域取决于声明时,而不取决于调用时。
【例2】
function prev(){
var leg = "Alexis";
var arsenal_leg = function(){
return leg;
}
return arsenal_leg; //前半赛季阿森纳的大腿是桑切斯
}
function after(){ //来到了后半赛季
var leg = "Giroud";
var arsenal_leg = prev(); //调用prev(),arsenal_leg 函数来到了后半赛季
alert(arsenal_leg());
}
after(); //谁是大腿
</script>
弹出:Alexis
【分析】:
line:14 执行 after();
line:12 alert arsenal_leg();
line:11 aesenal_leg() = prev();
line:6 return arsenal_leg();
line:3 函数声明 return leg;【真正执行的函数】
line:2 var leg = "Alexis";【作用域要从 函数声明之时寻找】
【例3】闭包计数器,多人开发 js 程序,需要一个全局的计数器
解决方案① 设立一个全局变量
window.cnt = 0;
调用 ++window.cnt
这个方案可行但是,污染了全局变量;其次比人的程序中,也可能会含有 window.cnt = ***; 则该计数器就会被损坏(所以要避免使用全局变量)
方案② 使用闭包维护一个别人污染不到的变量做技术器
<script>
function counter(){
var cnt = 0;
var cnter = function (){
return ++cnt;
}
return cnter;
}
var inc = counter();
alert(inc());
alert(inc());
alert(inc());
</script>
弹出 3 次,分别是:1,2,3
简化上面的程序:
<script>
var inc = (function counter(){
var cnt = 0;
return function (){
return ++cnt;
}
})();
alert(inc());
alert(inc());
alert(inc());
</script>
再改版上面的程序(inc 依然是全局变量。在工作中,一般避免全局污染或冲突):
方案 ① 统一放在一个全局对象上,如 jQuery -> $
<script>
$ = {};
//模拟jQuery
$.cnt = (function(){
var cnt = 0;
return function cnter(){
return ++cnt;
}
})();
alert($.cnt());
alert($.cnt());
alert($.cnt());
</script>
方案 ② 每个人使用自己的命名空间(其实就是把自己的变量、函数都放在一个对象里)
<script>
var Dee = {}; //一般用姓名做命名空间
Dee.cnt = (function(){
var cnt = 0;
return function cnter(){
return ++cnt;
}
})();
alert(Dee.cnt());
alert(Dee.cnt());
alert(Dee.cnt());
</script>
【例4】要求:点击 4 个 li,分别弹出 0,1,2,3
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Document</title>
</head>
<body>
<ul>
<li>阿森纳</li>
<li>切尔西</li>
<li>曼城</li>
<li>利物浦</li>
</ul>
</body>
<script> for(var i = 0, lis = document.getElementsByTagName("li"), len = lis.length; i < len; i++){
lis[i].onclick = (function(i){
return function(){
alert(i);
}
})(i);
} </script>
</html>
【错误的写法】(会弹出 4,4,4,4):
<script>
for(var i = 0, lis = document.getElementsByTagName("li"), len = lis.length; i < len; i++){
lis[i].onclick = function(){
alert(i);
}
}
</script>
【错误的写法分析】:
(回答者杨志):
" 这个for循环会立即执行完毕,那么当onclick触发时,inner function查找变量 i 时,会在AO+scope中找,AO中没有,scope中的变量i已经成为 link.length. ",
在运用了闭包之后,
" 这时,如果inner function被触发,他会从自己的AO以及scope(outer function的AO 和 window scope)中找寻变量i. 可以看到outer function的AO中已经包
含了i,而且对于这个for循环,会有对应有N个(function(){})() 被创建执行。所以每个inner function都有一个特定的包含了变量 i 的outer function。这样
就可以顺利输出0,1,2,3 。
结论: 我们可以看到,闭包其实就是因为Scope产生的,所以,广义上来讲,所有函数都是闭包。"
" 所谓的异步函数,是包括ajax,事件触发,setTimeout , setInterval等回调的函数。
当解析文档流时如果遇到js代码,会立即执行。如果这期间有异步函数被触发,会被存放在队列结构中。当之前的代码执行完毕后,浏览器查看此队列,如有待执行
函数,则执行;没有则等待触发。
所以,那个i 值会成为最大的那个。因为for循环在异步函数被执行前,已经跑完了。"
Javascript 笔记与总结(1-5)闭包的更多相关文章
- [Effective JavaScript 笔记]第27条:使用闭包而不是字符串来封装代码
函数是一种将代码作为数据结构存储的便利方式,代码之后可以被执行.这使得富有表现力的高阶函数抽象如map和forEach成为可能.它也是js异步I/O方法的核心.与此同时,也可以将代码表示为字符串的形式 ...
- [Effective JavaScript 笔记]第3章:使用函数--个人总结
前言 这一章把平时会用到,但不会深究的知识点,分开细化地讲解了.里面很多内容在高3等基础内容里,也有很多讲到.但由于本身书籍的篇幅较大,很容易忽视对应的小知识点.这章里的许多小提示都很有帮助,特别是在 ...
- [Effective JavaScript 笔记]第28条:不要信赖函数对象的toString方法
js函数有一个非凡的特性,即将其源代码重现为字符串的能力. (function(x){ return x+1 }).toString();//"function (x){ return x+ ...
- Jquery和Javascript 实际项目中写法基础-闭包 (2)
一.什么是闭包? 概念性的我就不去百度了,感兴趣的可以自己去搜下,我自己的理解,闭包就是一个封装的包,相当于类的概念,把乱七八糟的的东西封装到一起,然后统一使用一个对象来调用,实现代码部分对外开放,部 ...
- [Effective JavaScript 笔记] 第4条:原始类型优于封闭对象
js有5种原始值类型:布尔值.数字.字符串.null和undefined. 用typeof检测一下: typeof true; //"boolean" typeof 2; //&q ...
- [Effective JavaScript 笔记] 第5条:避免对混合类型使用==运算符
“1.0e0”=={valueOf:function(){return true;}} 是值是多少? 这两个完全不同的值使用==运算符是相等的.为什么呢?请看<[Effective JavaSc ...
- 从头开始学JavaScript 笔记(一)——基础中的基础
原文:从头开始学JavaScript 笔记(一)--基础中的基础 概要:javascript的组成. 各个组成部分的作用 . 一.javascript的组成 javascript ECMASc ...
- 【原】javascript笔记之Array方法forEach&map&filter&some&every&reduce&reduceRight
做前端有多年了,看过不少技术文章,学了新的技术,但更新迭代快的大前端,庞大的知识库,很多学过就忘记了,特别在项目紧急的条件下,哪怕心中隐隐约约有学过一个方法,但会下意识的使用旧的方法去解决,多年前ES ...
- 【译】学习JavaScript中提升、作用域、闭包的终极指南
这似乎令人惊讶,但在我看来,理解JavaScript语言最重要和最基本的概念是理解执行上下文.通过正确学习它,你将很好地学习更多高级主题,如提升,作用域链和闭包.考虑到这一点,究竟什么是"执 ...
- JavaScript笔记目录
JavaScript笔记目录 一.JavaScript简介 二.在HTML中使用JavaScript ...持续更新中,敬请期待
随机推荐
- (八)STM32的CAN模块实验
bxCAN是基本扩展CAN(Basic Extended CAN)的缩写,它支持CAN协议2.0A和2.0B.它的设计目标是,以最小的CPU负荷来高效处理大量收到的报文.它也支持报文发送的优先级要求( ...
- 蓝桥杯 算法训练 区间k大数查询(水题)
算法训练 区间k大数查询 时间限制:1.0s 内存限制:256.0MB 问题描述 给定一个序列,每次询问序列中第l个数到第r个数中第K大的数是哪个. 输入格式 第一行包含一个数n,表示序列长度. ...
- mac平台下面nodejs环境搭配
方法一(pkg方式:下载链接http://nodejs.org/dist/v0.12.4/node-v0.12.4.pkg): 官网:https://nodejs.org/download/ 例子: ...
- vijos 1028 LIS *
链接:点我 #include<cstdio> #include<iostream> #include<algorithm> #include<cstring& ...
- 关于Android开发中的证书和密钥等问题
关于Android开发中的证书和密钥等问题 引言 除了Android发布应用签名时需要用到证书外,在进行google Map Api开发和Facebook SDK API开发等时都需要申请API Ke ...
- Android ActionBar 一步一步分析 (转)
原文摘自:http://blog.csdn.net/android2me/article/details/8874846 1.Action Bar 介绍 我们能在应用中看见的actionbar一般就是 ...
- POJ 2482 Stars in Your Window 线段树扫描线
Stars in Your Window Description Fleeting time does not blur my memory of you. Can it really be 4 ...
- Gym 100463A Crossings 逆序对
Crossings Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100463 Description ...
- loj 1154(最大流+枚举汇点)
题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=26868 思路:拆点,容量为最多能跳的步数,然后设立一个超级源点,源 ...
- SSH 框架学习之初识Java中的Action、Dao、Service、Model-收藏
SSH 框架学习之初识Java中的Action.Dao.Service.Model-----------------------------学到就要查,自己动手动脑!!! 基础知识目前不够,有感性 ...