我知道
unset($a)
是會釋放記憶體空間(PS:排除被引用情況)
那麼如果我在一個迴圈中使用了一個比如$key
這個變數那麼當第二次
迴圈的時候$key
會變成1,那麼這時候感覺是$key
被重寫了。所以這個時候不顯式的使用unset的話,上一個$key還存在嗎?
回複內容:
我知道unset($a)
是會釋放記憶體空間(PS:排除被引用情況)
那麼如果我在一個迴圈中使用了一個比如$key
這個變數那麼當第二次
迴圈的時候$key
會變成1,那麼這時候感覺是$key
被重寫了。所以這個時候不顯式的使用unset的話,上一個$key還存在嗎?
大部分回答並不盡如人意,大家貌似都把PHP當成編譯性語言來解讀了。不過結果恰恰相反,PHP是指令碼語言,其特性和編譯性語言是有差別的。
簡單說說,PHP的變數依賴於一個內部實現 symbol_table
符號表,而符號表的基礎實現是 HashTable
,也就是和PHP數組的基礎實現是一致的。真是因為符號表的存在,讓我們可以使用global
標記全域變數,用如compact
等函數直接從當前符號表中拉出變數出來。
那在談談題主說的unset($a)
會不會馬上釋放空間,答案是否定的,unset
支援從符號表中把名字為a
的這個元素刪掉了(只是標記這塊空間又可用了,而不是釋放空間)。
再說迴圈中重複更新$key
這種情況,因為更新的是相同名字的變數,所以在符號表中他們是同一個元素,更新時就會更新相同的位置,之前元素的值就馬上被覆蓋了。
再說說申明了新的變數記憶體就會增加這個問題,答案是不確定。這是符號表基於 HashTable
實現的特性所致, HashTable
並不是增加一個元素就申請一個元素的記憶體,而是一次申請多個元素的記憶體(只是這些位置標記是未使用),而當 HashTable
被塞滿時,再去申請新的多個元素的記憶體。也就是說,當我們申明或者賦值一個變數時,如果它不在符號表中,PHP會將它加入到符號表裡,而如果這時候符號表沒滿,那會採用符號表中已申請而未使用的記憶體,如果符號表剛好的滿的,則會申請新的記憶體出來存放,而新的記憶體不僅僅只有這個變數需要的記憶體這麼小。
還真不好回答...
當 $key 未定義的時候是不佔據記憶體的
當 $key = 1 的時候$key被分配記憶體,1存在這裡
當 $key = 2 的時候,因為$key已經被分配記憶體了,所以不會再次分配,2存進去
沒有記錄。
我是這麼認為的。
我記得曾經在哪裡看過一篇文章說不建議二次賦值,效能問題
foreach的$key
值就相當於for迴圈裡的$i
下標,不過和for不同的是foreach每迴圈一次指標下移;
深入解析php中的foreach問題
$key 在 foreach 的時候每次會賦值