zoukankan      html  css  js  c++  java
  • 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
  • 相关阅读:
    c++ 面试注意的问题
    非root用户 gcc安装
    爱挑剔的acm程序员 acmer
    Add Digits
    Rectangle Area
    Contains Duplicate III 下标范围<=k 值范围<=t
    ruby第一次实践 ”hello world“
    判断一个 int 向量里是否有相同的数 并且距离在k之内 (2)
    判断一个 int 向量里是否有相同的数(1)
    字符串同构
  • 原文地址:https://www.cnblogs.com/iflygofy/p/11359803.html
Copyright © 2011-2022 走看看