js中那些让你不知所措的闭包
来源:互联网 发布:java事务怎么实现 编辑:程序博客网 时间:2024/06/12 00:27
1.先谈谈JS中的作用域:
在函数外部自然无法读取函数内的局部变量。
Js代码
function f1(){
var n=999;
}
alert(n); // error
这里有一个地方需要注意,函数内部声明变量的时候,一定要使用var命令。如果不用的话,你实际上声明了一个全局变量!
Js代码
function f1(){
n=999;
}
f1();
alert(n); // 999
2.如何从外部读取局部变量?
出于种种原因,我们有时候需要得到函数内的局部变量。但是,前面已经说过了,正常情况下,这是办不到的,只有通过变通方法才能实现。
那就是在函数的内部,再定义一个函数。
Js代码
function f1(){
n=999;
function f2(){
alert(n); // 999
}
}
在上面的代码中,函数f2就被包括在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1 就是不可见的。这就是Javascript语言特有的“链式作用域”结构(chain scope),
子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。
既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!
Js代码
function f1(){
n=999;
function f2(){
alert(n);
}
return f2;
}
var result=f1();
result(); // 999
这时我们就可以引入闭包了.....
闭包定义:官方定义:在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,
是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,
即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是
由函数和与其相关的引用环境组合而成的实体。
自己的理解
能够读到其他函数内部的变量或函数的函数
所以它闭的就是其他函数内部的变量,或函数;也就是我们所说的自由变量和引用环境;(闭的是什么)
function out(){
var a=2;
var b=3;
function inner(){
a++;
b++;
}
return function ins(){
inner();
}
}
var result=out();
result();
其中的inner中的a和b,就是自由变量;而inner(){}来说的话就是引用环境
由图可以知闭的是a和b还有inner()
在我们理解闭包后可以来看看这个例子:
一个糟糕的例子
document.getElementsByTagName('button')[0].onclick=function(){
var inp=document.getElementsByTagName('input');
for(var i=0;i<inp.length;i++){
inp[i].onclick=function(){
alert(i);
}
}
}
我想大家和我一样也在疑问为什么alert()出来的都是5嘞,其实我们用刚刚的闭包解释一下不难得出答案;
inp[i]中的i,相当于外部函数的自由变量,一直随着i++而变化;
而onclick()事件只有点击的时候才alert()而此时的i已等于5了;故有此结果
- js中那些让你不知所措的闭包
- 谈谈Js闭包的那些事儿
- Android中那些让你相见恨晚的类-接口-方法
- Android中那些让你相见恨晚的类-接口-方法
- Android Studio中那些让你相见恨晚的操作集锦
- 让你分分钟学会 JS 闭包
- 干货分享:让你分分钟学会JS闭包
- 让你详细了解js闭包问题
- XCODE7让我不知所措
- js闭包执行结束后的那些事
- 计算机中那些事儿(六):让你的计算机飞一般的感觉
- 职场中那些能够让你在困境中突围的能力
- Android开发中,那些让你觉得相见恨晚的方法、类或接口
- Android 开发中,那些让你觉得相见恨晚的方法、类或接口
- Android开发中,那些让你觉得相见恨晚的方法、类或接口
- Android开发中,那些让你觉得相见恨晚的方法、类或接口
- Android开发中,那些让你觉得相见恨晚的方法、类或接口
- Android开发中,那些让你相见恨晚的方法、类或接口
- 快速排序
- gsoap使用总结
- Python装饰器
- java 网络客户机与服务器的交互
- 搜狗浏览器的技巧
- js中那些让你不知所措的闭包
- Microsoft VBScript 运行时错误 错误 '800a0046' 没有权限
- Codeforces Round #442 (Div. 2) C. Slava and tanks
- spring boot——logback基本配置
- 图库
- 花生壳远程桌面连接
- C++ string 字符串查找匹配
- LaTex(PART XIII)\newcommand自定义命令
- 简要总结良好的C++编程习惯