看了這個回答似乎也不是瞭解的很透徹啊
我是學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裡面其實是不支援閉包的,匿名內部類看起來跟閉包差不多,但實際上不管是功能性還是實現層面都不能算是閉包。因為:
從功能上,匿名內部類裡面訪問的變數必須是final(Java8隱含聲明final)
實現上,匿名內部類裡訪問的final變數值其實是從外面被拷貝進去了,所以其並不能真正訪問到之前的範圍,這也是為什麼必須是final的原因。
所謂“閉包”,指的是一個擁有許多變數和綁定了這些變數的環境的運算式(通常是一個函數),因而這些變數也是該運算式的一部分。
在Javascript中建立一個閉包來解釋閉包最好不過:
function a(){var i=0;function b(){alert(++i);}return b;}var c=a();c();
函數b嵌套在函數a內部;函數a返回函數b。
這樣在執行完var c=a( )後,變數c實際上是指向了函數b,再執行c( )後就會彈出一個視窗顯示i的值(第一次為1)。這段代碼其實就建立了一個閉包,因為函數a外的變數c引用了函數a內的函數b,也就是說:當函數a的內建函式b被函數a外的一個變數引用的時候,就建立了一個閉包。
其實按照閉包的一般寫法形式,簡單的來說就是 函數裡面又嵌套了函數。在團隊開發中,為了防止命名衝突,我們一般會把相應的代碼用閉包的形式包裹起來,以避免暴露在全域範圍下面。但是有個不好的地方是其內部變數不會被立馬回收,有記憶體溢出的風險。