好程序员技术分析JavaScript闭包特性详解

发布时间:2025-08-31 21:49:29 作者:益华网络 来源:undefined 浏览量(0) 点赞(0)
摘要:好程序员

好程序员 技术分析 JavaScript 闭包特性详解 今天来总结一下 js 闭包 的那些事,以及遇到的坑和解决方法,希望对你有所帮助。

是的,没看错标题,重要的事情要说三篇, JavaScript 闭包。

首先先简要总结闭包特性:

·  函数的局部变量在函数返回之后仍然可用

·  栈上的内存空间在函数返回之后仍在存在,不被回收

给个例子。下面这段代码会返回一个函数的引用:

function sayHello2(name) {

    var text = Hello + name; // Local variable

    var sayAlert = function() { alert(text); }

    return sayAlert;

}

say2 = sayHello2(Bob);

say2(); // alerts "Hello Bob"

对于这段代码, C 程序员可能会认为 sayAlert say2 一样,都是指向一个函数的指针。但实际上它俩有一个重要区别: JavaScript 中,你可以认为一个函数的指针变量同时拥有两个指针。一个指向这个函数,另一个隐藏的指针指向一个闭包。

重点在于你的函数内是否引用的外部变量。

JavaScript 中,如果你在一个函数内定义一个新的函数,那么这个新的函数就是一个闭包。 对于 C 或者其他高级语言,函数执行结束并返回之后,它所占用的栈空间将被释放回收。函数内定义的局部变量将不再可用。但在 JavaScript 中,并不这样。如上所示,函数执行结束后,它所占用的栈空间并不会被全部回收。

上面是基本理论。更进一步,再来一个例子:

function say667() {

    // Local variable that ends up within closure

    var num = 666;

    var sayAlert = function() { alert(num); }

    num++;

    return sayAlert;

}

var sayNumber = say667();

sayNumber(); // alerts 667

这个例子说明:闭包中使用的函数局部变量并非是值拷贝,而是引用。 say667() 执行结束之后 number 所在的那块内存的值为 667 ,而 sayNumber() 是在 say667() 执行结束之后才执行,当它访问 number 所在的内存时,结果自然也是 667

再进一步,看看用 closure 时易发生的错误的例子:

function buildList(list) {

    var result = [];

    for (var i = 0; i < list.length; i++) {

        var item = item + list[i];

        result.push( function() {alert(item + + list[i])} );

    }

    return result;

}

function testList() {

    var fnlist = buildList([1,2,3]);

    // Using j only to help prevent confusion -- could use i.

    for (var j = 0; j < fnlist.length; j++) {

        fnlist[j]();

    }

}

时刻保持清醒:变量是在内存里的,闭包使用的是内存的引用而不是那块内存的值拷贝。

当你在循环中定义函数(闭包)的时候得小心,它可能并不像你最开始想的那样工作。关键有两个:

·  子函数使用的是外部函数的局部变量的引用。

·  循环内只是 定义 了子函数,并没有 执行 这个字函数。

最后,来一个最抽象的例子:

function newClosure(someNum, someRef) {

    // Local variables that end up within closure

    var num = someNum;

    var anArray = [1,2,3];

    var ref = someRef;

    return function(x) {

        num += x;

        anArray.push(num);

        alert(num: + num +

            \nanArray + anArray.toString() +

            \nref.someVar + ref.someVar);

      }

}

obj = {someVar: 4};

fn1 = newClosure(4, obj);

fn2 = newClosure(5, obj);

fn1(1); // num: 5; anArray: 1,2,3,5; ref.someVar: 4;

fn2(1); // num: 6; anArray: 1,2,3,6; ref.someVar: 4;

obj.someVar++;

fn1(2); // num: 7; anArray: 1,2,3,5,7; ref.someVar: 5;

fn2(2); // num: 8; anArray: 1,2,3,6,8; ref.someVar: 5;

这个例子说明,闭包的创建时机是在函数被调用的时候。每次函数调用都会生成一个新的闭包,也就是一块新的内存区域。因为函数每次调用都会新分配一块栈内存,这是一回事。

最后我自己来总结一下闭包:

·  函数的局部变量在其他地方被引用

·  闭包有两种基本情况:闭包的返回值是一个函数,它其中使用了该闭包的局部变量;闭包内定义了内部函数,内部函数引用了闭包的局部变量

·  每次函数调用,都会生成一个新的闭包,分配新的内存

实例:(滑过 tab

window.onload= function(){

var tits = $(#tabTit1 li);

var cons = $(#tabCon1 .con);

var len = cons.length;

var liChange = function(){

for(var n=0;n<len;n++){

tits[n].className = tits[n].className.replace(/\s*cur/g,);

cons[n].className = cons[n].className.replace(/\s*cur/g,);

}

}

for(var i = 0; i<tits.length; i++){

tits[i].i = i;

tits[i].onmouseover = function(){

liChange();

cons[this.i].addClass(cur);

tits[this.i].addClass(cur);

}

}

};

二维码

扫一扫,关注我们

声明:本文由【益华网络】编辑上传发布,转载此文章须经作者同意,并请附上出处【益华网络】及本页链接。如内容、图片有任何版权问题,请联系我们进行处理。

感兴趣吗?

欢迎联系我们,我们愿意为您解答任何有关网站疑难问题!

您身边的【网站建设专家】

搜索千万次不如咨询1次

主营项目:网站建设,手机网站,响应式网站,SEO优化,小程序开发,公众号系统,软件开发等

立即咨询 15368564009
在线客服
嘿,我来帮您!