闭包(超详解,看这一篇就够了)

闭包(超详解,看这一篇就够了)一 函数复习结论 闭包找到的是同一地址中父级函数中对应变量最终的值 1 概念闭包函数 声明在一个函数中的函数 叫做闭包函数

大家好,欢迎来到IT知识分享网。

一、函数复习

结论:闭包找到的是同一地址中父级函数中对应变量最终的值

1、概念
闭包函数:声明在一个函数中的函数,叫做闭包函数。

闭包:内部函数总是可以访问其所在的外部函数中声明的参数和变量,即使在其外部函数被返回(寿命终结)了之后。

2、特点
  让外部访问函数内部变量成为可能;

  局部变量会常驻在内存中;

  可以避免使用全局变量,防止全局变量污染;

  会造成内存泄漏(有一块内存空间被长期占用,而不被释放)

3、闭包的创建:­­­
闭包就是可以创建一个独立的环境,每个闭包里面的环境都是独立的,互不干扰。闭包会发生内存泄漏,每次外部函数执行的时 候,外部函数的引用地址不同,都会重新创建一个新的地址。但凡是当前活动对象中有被内部子集引用的数据,那么这个时候,这个数据不删除,保留一根指针给内部活动对象。

闭包内存泄漏为: key = value,key 被删除了 value 常驻内存中; 局部变量闭包升级版(中间引用的变量) => 自由变量;
 

1.1 函数的定义和调用

1.1.1函数的定义方式

方式1 函数声明方式 function fn(){}

方式2 函数表达式(匿名函数) var fn = function(){}

方式3 new Function() var fn = new Function(‘参数1’,’参数2’…, ‘函数体’)

  • 注意 Function 里面参数都必须是字符串格式
  • 第三种方式执行效率低,也不方便书写,因此较少使用
  • 所有函数都是 Function 的实例(对象)
  • 函数也属于对象

1.1.2函数的调用

普通函数的调用—>直接调用

function fn() { console.log('人生的巅峰'); } fn();

对象的方法—>对象调用

var o = { sayHi: function() { console.log('人生的巅峰'); } } o.sayHi()

构造函数–>new 调用

function Star(name,age) { this.name = name; this.age = age; }; new Star();

立即执行函数(IIFE)

(function () { })();

二、高阶函数

高阶函数是对其他函数进行操作的函数,它接收函数作为参数或将函数作为返回值输出。

  1. 接收函数作为参数—>回调函数
  2. 函数作为返回值输出

2.1 变量作用域复习

变量根据作用域的不同分为两种:全局变量和局部变量。

  • 函数内部可以使用全局变量。
  • 函数外部不可以使用局部变量。
  • 当局部函数执行完毕,本作用域内的局部变量会销毁。

问:如何从外部读取局部变量?

出于种种原因,我们有时候需要得到函数内的局部变量。但是,前面已经说过了,正常情况下,这是办不到的,只有通过变通方法才能实现。

那就是在函数的内部,再定义一个函数。

function f1(){     var n=999;     function f2(){       alert(n); // 999     } f2(); } f1(); 

2.2 什么是闭包

闭包(closure)指有权访问另一个函数作用域中变量的函数。(官方说法)

简单理解就是 ,定义在一个函数内部的,它可以访问到外部函数的局部变量

function init() { var name = "Mozilla"; // name 是一个被 init 创建的局部变量 function displayName() { // displayName() 是内部函数,一个闭包 alert(name); // 使用了父函数中声明的变量 } displayName(); } init(); 看重点:displayName() 没有自己的局部变量。 然而,因为它可以访问到外部函数的变量,所以 displayName() 可以使用父函数 init() 中声明的变量 name 。 在一些编程语言中,一个函数中的局部变量仅存在于此函数的执行期间。 一旦 init() 执行完毕,你可能会 认为 name 变量将不能再被访问。

2.3 闭包的特点

1. 函数嵌套

2. 内部函数使用外部函数作用域内的变量

闭包的经典案例: function foo() { var local = 1 function bar() { return ++local } return bar } var func = foo() console.log(func());

2.4 案例讲解-点击li输出索引号

for (var i = 0; i < lis.length; i++) { // 利用for循环创建了4个立即执行函数 // 立即执行函数也成为小闭包因为立即执行函数里面的任何一个函数都可以使用它的i这变量 (function(i) { lis[i].onclick = function() { console.log(i); } })(i); }

2.5 案例讲解-定时器中的闭包

for (var i = 0; i < lis.length; i++) { (function(i) { setTimeout(function() { console.log(lis[i].innerHTML); }, 3000) })(i); }

2.6 面试题

function foo() {

var i = 0;

return function () {

console.log(i++);

}

}

var f1 = foo();

var f2 = foo();

f1();

f1();

f1();

f1();

f2();

f2();

f2();

f2();

f2();

2.7 闭包的优缺点

当函数执行完毕,本作用域内的局部变量会销毁。

闭包的优点:

  • 1.能够读取函数内部的变量
  • 2.让这些变量一直存在于内存中,不会在调用结束后,被垃圾 回收机制回收

闭包的缺点:

  • 由于闭包会使函数中的变量保存在内存中,内存消耗很大,所以不能滥用闭包

免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://haidsoft.com/121245.html

(0)
上一篇 2025-10-24 19:33
下一篇 2025-10-24 19:45

相关推荐

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注微信