目錄
閉包? " >什麼是閉包
為啥要這樣寫?
常見的陷阱
總結一下
首頁 web前端 js教程 帶你一分鐘理解 JavaScript 閉包的圖文程式碼詳解

帶你一分鐘理解 JavaScript 閉包的圖文程式碼詳解

Mar 08, 2017 pm 02:14 PM

先看一段程式碼:

function a(){
    var n = 0;
    function inc() {
        n++;
        console.log(n);
    }
    inc();  
    inc(); 
}
a(); //控制台输出1,再输出2
登入後複製

簡單吧。再來看一段程式碼:

function a(){
    var n = 0;
    this.inc = function () {
        n++; 
        console.log(n);
    };
}
var c = new a();
c.inc();    //控制台输出1
c.inc();    //控制台输出2
登入後複製

簡單吧。

什麼是閉包?這就是閉包!

有權存取另一個函數作用域內變數的函數都是閉包。這裡 inc 函數存取了建構函式 a 裡面的變數 n,所以形成了一個閉包。

再來看一段程式碼:

function a(){
    var n = 0;
    function inc(){
       n++; 
       console.log(n);
    }
    return inc;
}
var c = a();
c();    //控制台输出1
c();    //控制台输出2
登入後複製

看看是怎麼執行的:

var c = couter(),這句話couter()回傳的是函數inc,那這句話等同於var c = inc;

c(),這句話等同於inc();  注意,函數名稱只是一個識別(指向函數的指標),而()才是執行函數。

後面三句翻譯過來就是:  var c = inc;  inc();  inc();,跟第一段程式碼有差嗎? 沒有。

什麼是閉包?這就是閉包!

所有的教科書教學上都喜歡用最後一段來說明閉包,但我覺得這將問題複雜化了。這裡面回傳的是函數名,沒看過譚浩強C/C++程式設計的同學可能一下子沒反應出帶不帶()的差別,也就是說這種寫法自帶一個陷阱。雖然這種寫法更顯高大上,但我還是喜歡將問題單一化,看看代碼 1 和代碼 2,你還會糾結函數的調用,你會糾結 n 的值嗎?

為啥要這樣寫?

我們知道,js的每個函數都是一個個小黑屋,它可以獲得外界訊息,但是外界卻無法直接看到裡面的內容。將變數n 放進小黑屋裡,除了inc 函數之外,沒有其他辦法能接觸到變數n,而且在函數a 外定義同名的變數n 也是互不影響的,這就是所謂的增強“封裝性” 。

而之所以要用return 返回函數來識別inc,是因為在a 函數外部無法直接呼叫inc 函數,所以return inc 與外部聯繫起來,代碼2 中的this 也是將inc 與外部聯繫起來而已。

常見的陷阱

看看這個:

function createFunctions(){
    var result = new Array();
    for (var i=0; i < 10; i++){
        result[i] = function(){
            return i;
        };
    }
    return result;
}
var funcs = createFunctions();
for (var i=0; i < funcs.length; i++){
    console.log(funcs[i]());
}
登入後複製

乍一看,以為輸出 0~9 ,萬萬沒想到輸出10個10?

這裡的陷阱就是:函數帶()才是執行函數! 單純的一句 var f = function() { alert(‘Hi’); }; 是不會彈出視窗的,後面接著一句 f(); 才會執行函數內部的程式碼。上面程式碼翻譯一下就是:

var result = new Array(), i;
result[0] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!
result[1] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!
...
result[9] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!
i = 10;
funcs = result;
result = null;

console.log(i); // funcs[0]()就是执行 return i 语句,就是返回10
console.log(i); // funcs[1]()就是执行 return i 语句,就是返回10
...
console.log(i); // funcs[9]()就是执行 return i 语句,就是返回10
登入後複製

為什麼只垃圾回收了 result,但卻不收了 i 呢? 因為 i 還在被 function 引用著啊。好比一個餐廳,盤子總是有限的,所以服務生會去巡台回收空盤子,但還裝著菜的盤子他怎麼敢收? 當然,你自己手動倒掉了盤子裡面的菜(=null),那盤子就會被收走了,這就是所謂的內存回收機制。

至於 i 的值怎麼還能保留,其實從文章開頭一路讀下來,這應該沒有什麼可以糾結的地方。盤子裡面的菜,吃了一塊不就該少一塊嗎?

總結一下

閉包就是一個函數引用另一個函數的變量,因為變數被引用所以不會被回收,因此可以用來封裝一個私有變數。這是優點也是缺點,不必要的閉包只會徒增記憶體消耗!另外使用閉包也要注意變數的值是否符合你的要求,因為他就像靜態私有變數。閉包通常會跟很多東西混搭起來,接觸多了才能加深理解,這裡只是開個頭說說基礎性的東西。


以上是帶你一分鐘理解 JavaScript 閉包的圖文程式碼詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
2 週前 By 尊渡假赌尊渡假赌尊渡假赌
倉庫:如何復興隊友
4 週前 By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒險:如何獲得巨型種子
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

C++ lambda 表達式中閉包的意思是什麼? C++ lambda 表達式中閉包的意思是什麼? Apr 17, 2024 pm 06:15 PM

C++ lambda 表達式中閉包的意思是什麼?

C++ Lambda 表達式如何實作閉包? C++ Lambda 表達式如何實作閉包? Jun 01, 2024 pm 05:50 PM

C++ Lambda 表達式如何實作閉包?

C++ 函式中閉包的優點和缺點是什麼? C++ 函式中閉包的優點和缺點是什麼? Apr 25, 2024 pm 01:33 PM

C++ 函式中閉包的優點和缺點是什麼?

解決閉包導致的記憶體洩漏問題 解決閉包導致的記憶體洩漏問題 Feb 18, 2024 pm 03:20 PM

解決閉包導致的記憶體洩漏問題

函數指標和閉包對Golang效能的影響 函數指標和閉包對Golang效能的影響 Apr 15, 2024 am 10:36 AM

函數指標和閉包對Golang效能的影響

閉包在 Java 中是如何實現的? 閉包在 Java 中是如何實現的? May 03, 2024 pm 12:48 PM

閉包在 Java 中是如何實現的?

PHP 函數的鍊式呼叫與閉包 PHP 函數的鍊式呼叫與閉包 Apr 13, 2024 am 11:18 AM

PHP 函數的鍊式呼叫與閉包

golang匿名函數和閉包的優缺點總結 golang匿名函數和閉包的優缺點總結 May 05, 2024 am 09:54 AM

golang匿名函數和閉包的優缺點總結

See all articles