05. 函数高级-闭包
01. JavaScript的运行机制
(1)所有同步任务都在主线程上执行,形成一个执行栈。
(2)主线程之外,还有一个“任务队列”,只要异步任务有了运行结果,就在“任务队列”之中放置一个事件。
(3)一旦“执行栈”中的所有同步任务执行完毕了,系统就会读取“任务队列”,看看里面有哪些事件。那些对应的异步任务,于是结束等待状态,进入执行栈,开始执行。
(4)主线程不断重复上面的三步。(事件循环)
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>00_引入</title> </head> <body> <div id="btn01"> <button>测试11</button> <button>测试12</button> <button>测试13</button> </div> <hr/> <div id="btn02"> <button>测试21</button> <button>测试22</button> <button>测试23</button> </div> <hr/> <div id="btn03"> <button>测试31</button> <button>测试32</button> <button>测试33</button> </div> <!-- 需求: 点击某个按钮, 提示"点击的是第n个按钮" --> <script type="text/javascript"> var btns01 = document.getElementById('btn01').getElementsByTagName('button'); var btns02 = document.getElementById('btn02').getElementsByTagName('button'); var btns03 = document.getElementById('btn03').getElementsByTagName('button'); console.log(btns01) // 有问题 JavaScript的运行机制 主线程上执行,形成一个执行栈,一旦“执行栈”中的所有同步任务执行完毕了,系统就会读取“任务队列” for(var i=0,length=btns01.length;i<length;i++) { var btn = btns01[i] console.log(btn) btn.onclick = function () { alert('第'+(i)+'个 ' + btn.innerHTML) } } // 解决一: 保存下标 for(var i=0,length=btns02.length;i<length;i++) { var btn02 = btns02[i] btn02.index = i btn02.onclick = function () { alert('第'+(this.index+1)+'个 ' + btns02[this.index].innerHTML ) } } // 解决办法二: 利用闭包 for(var i=0,length=btns03.length;i<length;i++) { (function (i) { var btn03 = btns03[i] btn03.onclick = function () { alert('第'+(i+1)+'个 ' + btn03.innerHTML ) } })(i) } </script> </body> </html>
02. 理解闭包
1. 如何产生闭包?
* 当一个嵌套的内部(子)函数引用了嵌套的外部(父)函数的变量(函数)时, 就产生了闭包
2. 闭包到底是什么?
* 使用chrome调试查看
* 理解一: 闭包是嵌套的内部函数(绝大部分人)
* 理解二: 包含被引用变量(函数)的对象(极少数人)
* 注意: 闭包存在于嵌套的内部函数中
3. 产生闭包的条件?
* 函数嵌套
* 内部函数引用了外部函数的数据(变量/函数)
function fn1 () { var a = 3 function fn2 () { console.log(a) } fn2() } fn1()
03. 常见的闭包
1. 将函数作为另一个函数的返回值
2. 将函数作为实参传递给另一个函数调用
// 1. 将函数作为另一个函数的返回值 function fn1() { var a = 2 function fn2() { a++ console.log(a) } return fn2 } var f = fn1() f() // 3 f() // 4 // 2. 将函数作为实参传递给另一个函数调用 function showMsgDelay(msg, time) { setTimeout(function () { console.log(msg) }, time) } showMsgDelay('hello', 1000)
04. 闭包的作用
1. 使用函数内部的变量在函数执行完后, 仍然存活在内存中(延长了局部变量的生命周期)
2. 让函数外部可以操作(读写)到函数内部的数据(变量/函数)
问题:
1. 函数执行完后, 函数内部声明的局部变量是否还存在?
2. 在函数外部能直接访问函数内部的局部变量吗?
function fun1() { var a = 3; function fun2() { a++; //引用外部函数的变量--->产生闭包 console.log(a); } return fun2; } var f = fun1(); //由于f引用着内部的函数-->内部函数以及闭包都没有成为垃圾对象 f(); // 4 间接操作了函数内部的局部变量 f(); // 5
05. 闭包的生命周期
1. 产生: 在嵌套内部函数定义执行完时就产生了(不是在调用)
2. 死亡: 在嵌套的内部函数成为垃圾对象时
function fun1() { //此处闭包已经产生 var a = 3; function fun2() { a++; console.log(a); } return fun2; } var f = fun1(); f(); f(); f = null //此时闭包对象死亡
06. 闭包的应用
自定义JS模块
* 具有特定功能的js文件
* 将所有的数据和功能都封装在一个函数内部(私有的)
* 只向外暴露一个包信n个方法的对象或函数
* 模块的使用者, 只需要通过模块暴露的对象调用方法来实现对应的功能
// 自定义模块1 function coolModule() { //私有的数据 var msg = 'message' var names = ['I', 'Love', 'you'] //私有的操作数据的函数 function doSomething() { console.log(msg.toUpperCase()) } function doOtherthing() { console.log(names.join(' ')) } //向外暴露包含多个方法的对象 return { doSomething: doSomething, doOtherthing: doOtherthing } }
var module = coolModule() module.doSomething() module.doOtherthing()
自定义模块2
// 自定义模块2 (function (window) { //私有的数据 var msg = 'atguigu' var names = ['I', 'Love', 'you'] //操作数据的函数 function a() { console.log(msg.toUpperCase()) } function b() { console.log(names.join(' ')) } window.coolModule2 = { doSomething: a, doOtherthing: b } })(window)
coolModule2.doSomething()
coolModule2.doOtherthing()
07. 闭包的缺点及解决
1. 缺点
* 函数执行完后, 函数内的局部变量没有释放, 占用内存时间会变长
* 容易造成内存泄露
2. 解决
* 能不用闭包就不用
* 及时释放
function fn1() { var a = 2; function fn2() { a++; console.log(a); } return fn2; } var f = fn1(); f(); // 3 f(); // 4 f = null // 释放
08. 面试题
// 说说它们的输出情况 //代码片段一 var name = "The Window"; var object = { name: "My Object", getNameFunc: function () { return function () { return this.name; }; } }; console.log(object.getNameFunc()()); //The Window //代码片段二 var name2 = "The Window"; var object2 = { name2: "My Object", getNameFunc: function () { var that = this; return function () { return that.name2; }; } }; console.log(object2.getNameFunc()()); // My Object
// 说说它们的输出情况 function fun(n, o) { console.log(o) return { fun: function (m) { return fun(m, n) } } } var a = fun(0) a.fun(1) a.fun(2) a.fun(3) //undefined,0,0,0 var b = fun(0).fun(1).fun(2).fun(3) //undefined,0,1,2 var c = fun(0).fun(1) c.fun(2) c.fun(3) //undefined,0,1,1