楊__羊羊
2019-03-04 16:12:48
js 權威指南-第六版-中 p184 實現閉包,有段話理解不了;全文是這樣:每次調用 js 函數時,都會為之創建一個新的變量對象用來保存局部變量,把這個對象添加至作用域鏈中。當函數返回時,就從作用域鏈中將這個綁定的變量對象刪除。如果不存在嵌套的函數,也沒有其他引用指向這個綁定對象,他就會被當做垃圾回收掉。如果定義了嵌套的函數,每個嵌套的函數都各自對應一個作用域鏈,并且這個作用域鏈指向一個變量綁定對象。但如果這些嵌套的函數對象在外部函數中保存下來,那么他們也會和所指向的變量綁定對象一樣當做垃圾回收。但是如果這個函數定義了嵌套的函數,并將它作為返回值返回或者存儲在某處的屬性里,這時就會有一個外部引用指向這個嵌套的函數,他就不會被當做垃圾回收,并且他所指向的變量綁定對象也不會被當做垃圾回收。我分成三段理解:1、每次調用 js 函數時,都會為之創建一個新的變量對象用來保存局部變量,把這個對象添加至作用域鏈中。當函數返回時,就從作用域鏈中將這個綁定的變量對象刪除。如果不存在嵌套的函數,也沒有其他引用指向這個綁定對象,他就會被當做垃圾回收掉??梢岳斫馍厦嬉欢危旅鎯啥尾惶宄?;2、如果定義了嵌套的函數,每個嵌套的函數都各自對應一個作用域鏈,并且這個作用域鏈指向一個變量綁定對象。但如果這些嵌套的函數對象在外部函數中保存下來,那么他們也會和所指向的變量綁定對象一樣當做垃圾回收。3、但是如果這個函數定義了嵌套的函數,并將它作為返回值返回或者存儲在某處的屬性里,這時就會有一個外部引用指向這個嵌套的函數,他就不會被當做垃圾回收,并且他所指向的變量綁定對象也不會被當做垃圾回收。結合高程,可以理解3、,因為換成代碼就是這樣:createComparisonFunction()函數在執行完畢后,其活動對象也不會被銷毀,因為匿名函數的作用域鏈仍然在引用這個活動對象。這個很好理解;但是2、比較模糊,換成代碼我覺得就是下面這樣,所以問題是:var compareNames = createComparisonFunction("name"); 執行完了后,createComparisonFunction()的活動對象會被銷毀么?回答1:會銷毀,因為按照 2、所說,但如果這些嵌套的函數對象在外部函數中保存下來,那么他們也會和所指向的變量綁定對象一樣當做垃圾回收。返回的 f 在 外部函數中保存下來了,所以會和所指向的變量綁定對象一樣當做垃圾回收?;卮?:不會銷毀,因為返回的 f 里依然保存著對 createComparisonFunction() 活動對象的引用啊,就是那個 propertyName ,所以 createComparisonFunction() 活動對象不會銷毀;所以哪個解釋是對的?還是有其他解釋?
2 回答

慕娘9325324
TA貢獻1783條經驗 獲得超4個贊
如果定義了嵌套的函數,每個嵌套的函數都各自對應一個作用域鏈,并且這個作用域鏈指向一個變量綁定對象。但如果這些嵌套的函數對象在外部函數中保存下來,那么他們也會和所指向的變量綁定對象一樣當做垃圾回收。
一共兩句話。前面那句只是說嵌套函數也有自己的作用域鏈和變量綁定對象,所以主要是后面那一句。這句話的意思是,外部函數可以在其變量綁定對象中保存嵌套函數的引用,但是如果沒有把嵌套函數返回給其他代碼,那么當外部函數銷毀的時候,其保存的嵌套函數也會同時被銷毀回收。
翻譯成代碼是這樣的:
function outer() {
var a = 3;
function inner() {
var b = 5;
}
// 這時候outer的變量綁定對象里面有a和inner兩個變量
// 所以當outer被回收的時候,a和inner也會被同時回收
...
}
第三段話則比上面這段話的意思更進了一步。
但是如果這個函數定義了嵌套的函數,并將它作為返回值返回或者存儲在某處的屬性里,這時就會有一個外部引用指向這個嵌套的函數,他就不會被當做垃圾回收,并且他所指向的變量綁定對象也不會被當做垃圾回收。
如果inner被outer作為返回值返回的話,那么就不僅僅是outer保有它的引用了,外面會有其他變量也保有它的引用。所以當outer被回收的時候,inner并不會回收。
添加回答
舉報
0/150
提交
取消