首頁  >  文章  >  後端開發  >  javascript - 面試問題:為什麼要用閉包?

javascript - 面試問題:為什麼要用閉包?

WBOY
WBOY原創
2016-12-01 00:26:031557瀏覽

看了這個回答似乎也不是了解的很透徹啊

我是學Java的。今天在面試的時候面試官提及匿名類,我說Java8裡面提供了Lamada式,在JS裡也有閉包這個概念。面試官問什麼是閉包?為什麼要用閉包?

  • 我說:用來控制訪問啊。內部可以存取外部,但是外部不能存取內部。

面試官似乎不是很滿意這樣的回答。不知各位大神對這個問題有什麼看法?

回覆內容:

看了這個回答似乎也不是了解的很透徹啊

我是學Java的。今天在面試的時候面試官提及匿名類,我說Java8裡面提供了Lamada式,在JS裡也有閉包這個概念。面試官問什麼是閉包?為什麼要用閉包?

  • 我說:用來控制訪問啊。內部可以存取外部,但是外部不能存取內部。

面試官似乎不是很滿意這樣的回答。不知各位大神對這個問題有什麼看法?

閉包,顧名思義,就是把饅頭變成包子~

饅頭全是麵粉,包上餡就成了包子

包子是帶餡的饅頭

閉包是自帶運作環境的函數

髮哥是自備背景音樂的男人~


有童鞋不懂「自備運行環境」的意思~

再舉例說一次吧~

碼農們都吃過泡麵吧~
它跟普通麵條有什麼差別呢?
就是 自帶調味包 。
調味包就是泡麵的烹飪環境。
它簡化了煮麵條的流程。讓用戶不必練就廚藝也能吃上美味的淚流滿面。

函數式程式設計的閉包,就是函數的調味包。
方便用戶呼叫函數。不必為了維護繁雜的外部狀態而煩惱。
例如python,就把閉包玩出了很多花樣:
靜態私有變數~
偏函數啦~
單參化~
裝飾器~
……

當你在使用這些功能的時候,其實就是在吃別人設定好調味包的「泡麵」。

其實我的理解:閉包的目的是用來擴大變數的作用域的。

立即執行函數和閉包有什麼關係

重點看下@邊城的理解,我的答案比較單薄,要是還感興趣,可以看下高程三對作用域鏈、閉包這段的闡述~

剛看的一篇文章 史上被罵最多的程式語言-JavaScript

樓主的回答並不準確,如果我是面試官我也不滿意。
簡單來說,閉包是指當函數被當成物件回傳時,如果夾帶了外部變數就形成了閉包。我非常認同那位比喻把饅頭加上餡變成包子的同學的回答,他雖是調侃成分居多,但理解的程度非常之深刻。

如果一個函數打包了外部變量,就可以給程式非常大的靈活性,你可以把閉包理解成輕量級的介面封裝,雖然對外都是這個函數(呼叫方式不變),但是因為之中的變數不一樣,就可以完成很多功能。這也就是那位同學說的自備運行環境的函數,自備背景音樂的男人,想想都很可怕。

如果你想還深入了解一點,可以參考我總結的一篇文章,詳解Python中的閉包,雖然程式語言不一樣,但是道理是一樣的。

你的回答是關於作用域的回答,不是關於閉包的。而這個回答也是屬於不嚴謹的回答。你根本解釋不清楚什麼是內部,什麼是外部。

我認為閉包是這樣的。當一個函數在定義它的作用域以外的地方被呼叫時,它所訪問的仍然是定義它時的作用域。這種現象稱為閉包。

具體用途有很多,常見的有創建私有屬性,函數柯里化等等。

------------分割線------------

我再補充一下,其實閉包的本質是靜態作用域。因為 JavaScript 沒有動態作用域,所以函數存取的都是定義時的作用域,所以閉包才得以實現。

其他答案裡說閉包是自帶運作環境的函數。但實際上,JavaScript 裡任何函數不都是自帶運作環境的函數嗎?有的人也因此認為所有的函數都是閉包。這當然也不算錯,但對理解閉包其實意義不大。因為你平常都是這麼使用函數的,即使你不知道什麼是閉包,也不會出什麼問題。只不過平常你可能沒有意識到全域作用域就是一個大閉包。

我們常見的閉包形式就是a 函數套 b 函數,然後 a 函數傳回 b 函數,這樣 b 函數在 a 函數以外的地方執行時,仍能存取 a 函數的作用域。其中「b 函數在 a 函數以外的地方執行時」這一點,才體現了閉包的真正的強大之處。

總之,閉包只是基於靜態作用域的一個程式設計技巧。從面試的角度來說,你要回答什麼是閉包,你首先得解釋什麼靜態作用域的特點,然後還必須要強調「b 函數在a 函數以外的地方執行時」這一點,才算是對閉包的完整回答。

lambda演算式只允許單輸入單輸出,所以lambda a, b: a + b等於lambda a: lambda b: a + b也就是currying。

簡單來說,閉包的定義是:函數能存取它被定義時的作用域。

所以,你說的存取控制之類的,只是閉包的一個應用場景而已,當然沒有回答什麼是閉包的問題,面試官自然就不滿意。

另外,Java裡面其實是不支援閉包的,匿名內部類別看起來跟閉包差不多,但其實不管是功能性還是實作層面都不能算是閉包。因為:

  1. 從功能上,匿名內部類別裡面存取的變數必須是final(Java8隱含聲明final)

  2. 實作上,匿名內部類別裡存取的final變數值其實是從外面被拷貝進去了,所以並不能真正存取到先前的作用域,這也是為什麼必須是final的原因。

所謂“閉包”,指的是一個擁有許多變數和綁定了這些變數的環境的表達式(通常是一個函數),因而這些變數也是該表達式的一部分。
在Javascript中創建一個閉包來解釋閉包最好不過:

<code>function a(){
var i=0;
function b(){
alert(++i);
}
return b;
}
var c=a();
c();</code>

函數b嵌套在函數a內部;函數a返回函數b。
這樣在執行完var c=a( )後,變數c其實是指向了函數b,再執行c( )後就會彈出一個視窗顯示i的值(第一次為1)。這段程式碼其實就創建了一個閉包,因為函數a外的變數c引用了函數a內的函數b,也就是說:當函數a的內部函數b被函數a外的一個變數引用的時候,就創建了一個閉包。

其實依照閉包的一般寫法形式,簡單的來說就是 函數裡面又嵌套了函數。在團隊開發中,為了防止命名衝突,我們一般會把對應的程式碼用閉包的形式包裹起來,以避免暴露在全域作用域下面。但是有個不好的地方是其內部變數不會被立刻回收,有記憶體溢出的風險。

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