關於php解構函式的一個有趣問題

傑楓Jeff發表於2015-08-22

  隨著物件導向程式設計的普遍展開,物件導向展現了其中很多有趣的問題。相信很多初學者學習php物件導向時會接觸兩個函式,建構函式與解構函式。建構函式似乎用的更多,解構函式用的較少(相對初學者有限程式設計經驗而言,筆者也是如此。)在功能上,建構函式在建立物件時呼叫,解構函式在物件銷燬時呼叫,都無需特意去呼叫,一頭一尾,倒也是前後照應。

  解構函式常常處理的事務是一些資源釋放的工作,比如前面有fopen(),這裡呼叫fclose(),前面有imagecreatefromjepg(),這裡呼叫imagedestory(),這些都是些常見的例子,當然不侷限於此。我們大可將其當做一個普通的會在物件銷燬或者指令碼執行完畢的時候執行的函式。

  囉嗦那麼多,還是儘早提出今天的主要的問題:

<?php
class
Test{ public function __destruct(){ echo "執行解構函式"; } } $test1=new Test; $test2=$test3=$test1;
unset($test1);
echo "<hr/>";

  這段指令碼執行結果是什麼?

  在回答這個問題之前,回看我上面標示的幾個字。我們可以理所應當的認為在輸出分隔線前 unset($test1) ,這樣會呼叫解構函式,輸出文字,至於 $test2,$test3 應該會在指令碼執行完畢呼叫解構函式。也就是說,在分割線上面,會輸出一段文字,分隔線下面會輸出兩段文字。 在這個時候,大可以小驕傲一下,畢竟自己懂得什麼時候呼叫解構函式。但現實真是如此麼?我們可以看一下執行結果。

  

  嘿,他喵的,怎麼就輸出了一句啊???

  其實我們忽略了一個重要的前提條件,就是物件的賦值預設的是引用賦值。這一點很多人沒有注意到,希望初學者能多多注意一下。

  那麼既然是引用賦值,結合我們對普通變數的理解,我們很快想到,三個變數名指向同一塊儲存地址。那麼既然如此的話, unset($test1) 起到的是什麼作用???破壞變數指向儲存地址還是破壞儲存地址儲存的內容?

  瞭解unset()函式用法的請直覺跳過本段。

  想著這蛋疼的問題,莫不如去檢視手冊。

  同樣是傳遞引用,毀掉的僅僅是變數名指向儲存地址。結合平常unset()的作用,我們可以這麼描述,當多個變數名或者物件名指向一塊儲存地址時,unset()函式的作用僅僅是銷燬變數名和儲存地址的指向而已,當僅有一個變數名或者物件名,unset銷燬的是指定的儲存地址上的內容。

  我們可以想象真實的儲存內容是一臺電視。多個人(多個變數名或物件名)在看一臺電視。unset()後,一個人不看了,離開了,電視還開著。當只有一個人看電視的時候,unset()後,人離開的時候,要把電視關了 ,也就是釋放佔用的儲存空間。對這部分內容感興趣的話也可以看一下《php 遞迴函式的三種實現方式》。

  

  好的,迴歸主題。 unset($test1) 後, 原來的物件還在。當輸出分割線後,指令碼執行完畢,呼叫解構函式。 因為物件只有一個,呼叫解構函式也只呼叫一次。輸出如上結果也就理所應當。

 

  其他幾個相關的有意思的問題:在程式中呼叫解構函式的方法還有很多。無論是設定物件為null,還是false,其餘的物件依舊不受影響。這和普通變數還是有區別的。(unset()函式的效果是一樣的)。有興趣的話可以試試。

  另外說一個比較囧的事情:我們都知道建構函式可以使用__construct(),卻忽視了同名建構函式。所以,大家還是留心一下。

 

相關文章