JS中的闭包和匿名函数自执行

1.闭包

闭包就是能够读取其他函数内部变量的函数。例如在 javascript 中,只有函数内部的子函数才能读取局部变量,所以闭包可以理解成“定义在一个函数内部的函数“。在本质上,闭包是将函数内部和函数外部连接起来的桥梁。
闭包通常用来创建内部变量,使得这些变量不能被外部随意修改,同时又可以通过指定的函数接口来操作;闭包就是有权限访问另一个函数内部作用域的变量的函数。

举个栗子

function foo() {
    var a = 10;
    function bar() {   //这个函数叫做闭包
        console.log(a);
    }
}
var foo = function () {
    var a = 10;
    return function () {
        console.log(a);
    };

}();     //return出来的是一个函数表达式,可以在后面接小括号直接调用。
foo();   //10  利用闭包可以在函数外部进行调用。

闭包有什么好处呢?
1.防止变量冲突。
2.拥有自己独立的变量作用域,让局部变量保持一个,避免全局变量的污染。

再举一个实用的栗子

var foo = function () {
    var a = 10;
    a++;
    console.log(a);
};
foo();  //11
foo();  //11
foo();  //11  每调用一次函数,a都会被重新赋值为10

我想让 a 值累加,又是一个局部变量,该怎么做呢?对了,闭包可以做到。

var foo = function () {
    var a = 10;
    return function () {
        a++;
        console.log(a);
    };
}();  //此时foo是函数执行完的返回值,即return后的闭包
foo();  //11
foo();  //12
foo();  //13

2.匿名函数自执行

匿名函数,顾名思义指的是没有名字的函数。(没有函数名)

function () {   //错误写法,会报错!默认情况下,程序认为没有名字的函数是函数声明写法
    console.log(123);
}();

可以通过以下方法转化为函数表达式
在 function 前面加!、+、-、~或者把函数用()包起来都可以将函数声明转换成函数表达式,我们一般用()把函数声明包起来。

(function () {
    console.log(123);
})();

可以配合闭包用来保存状态,下面是一道很经典的题。

for (var i = 0; i < 3; i++) {
    setTimeout(function () {
        console.log(i);  // 3 3 3
        //在执行到这一行时,发现匿名函数里没有i,然后向往外部作用域找,然后找到的其实是for循环执行完了的i,也就是2++,3
    }, 1000);
}
for (var i = 0; i < 3; i++) {
    (function (x) {
        setTimeout(function () {
            console.log(x);  // 0 1 2
        }, 1000);
    })(i)
    //i传给了x,并且锁在内存中,所以不会变
}
for (let i = 0; i < 3; i++) {    //let具有独立的块级作用域
    setTimeout(function () {
        console.log(i);  // 0 1 2
    }, 1000);
}
匿名函数,也称为拉姆达函数,是一种使用JavaScript函数的强大方式。以下总结了匿名函数的特点: 任何函数表达式从技术上说都是匿名函数,因为没有引用它们的确定的方式; 在无法确定如何引用函数的情况下,递归函数就会变得比较复杂; 递归函数应该始终使用arguments.callee来递归地调用自身,不要使用函数名--函数名可能会发生变化。 当在函数内部定义了其他函数时,就创建了闭包闭包有权访问包含函数内部的所有变量,原理如下: 在后台执行环境闭包的作用域链包含着它自己的作用域、包含函数的作用域和全局作用域; 通常,函数的作用域及其所有变量都会在函数执行结束后被销毁; 但是,当函数返回了一个闭包时,这个函数的作用域将会一直在内存保存到闭包不存在为止;使用闭包可以在JavaScript模仿块级作用域(JavaScript本身没有块级作用域的概念),要点如下: 创建并立即调用一个函数,这样既可以执行的代码,又不会在内存留下对该函数的引用; 结果就是函数内部的所有变量都会被立即销毁--除非将某些变量赋值给了包含作用域(即外部作用域)的变量。 闭包还可以用于在对象创建私有变量,相关概念和要点如下: 即使JavaScript没有正式的私有对象属性的概念,但可以使用闭包来实现公有方法,而通过公有方法可以访问在包含作用域定义的变量; 有权访问私有变量的公有方法叫做特权方法; 可以使用构造函数模式、原型模式来实现自定义类型的特权方法,也可以使用模块模式、增强的模块模式来实现单例的特权方法。 JavaScript匿名函数闭包都是非常有用的特性,利用它们可以实现很多功能。不过,因为创建闭包必须维护额外的作用域,所以过度使用它们可能会占用大量内存。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值