那些年我們錯過的魔術方法(Magic Methods)

NO IMAGE

PHP 物件的一個優勢是可以使用魔術方法,這些方法可以不需要修改外部程式碼而重寫一個類的預設行為,這使得PHP 語法有更少的冗餘性和更具有擴充套件性。這些方法很好識別,他們都是以雙下劃線(__)開始的。
比如: __construct(),__destruct(),__call(),__callStatic(),__get(),__set(),__isset(),__unset(),__sleep(),__wakeup(),__toString(),__invoke(),__set_state() 和 __clone() 等方法在 PHP 中被稱為”魔術方法”(Magic methods)。在命名自己的類方法時不能使用這些方法名,除非是想使用其魔術功能。

注意:

PHP 將所有以 __(兩個下劃線)開頭的類方法保留為魔術方法。所以在定義類方法時,除了上述魔術方法,建議不要以 __ 為字首。

1、__get、__set

這兩個方法是為在類和他們的父類中沒有宣告的屬性而設計的。

複製程式碼 程式碼如下:
__get(string $name)  //當呼叫一個未定義的屬性時訪問此方法;
__set(string $name, mixed $value)  //給一個未定義的屬性賦值時呼叫;

這裡的沒有宣告包括當使用物件呼叫時,訪問控制為proteced,private的屬性(即沒有許可權訪問的屬性)。

2、__isset、__unset

複製程式碼 程式碼如下:
__isset( $property ) //當在一個未定義的屬性上呼叫isset()函式時呼叫此方法;
__unset( $property ) //當在一個未定義的屬性上呼叫unset()函式時呼叫此方法;

3、__call、__callStatic

複製程式碼 程式碼如下: __call(string $name, array $arguments) //當呼叫一個未定義的方法是呼叫此方法。

這裡的未定義的方法包括沒有許可權訪問的方法。

複製程式碼 程式碼如下: __callStatic(string $name, array $arguments)

當在靜態方法中呼叫一個不可訪問方法(如未定義,或者不可見)時,__callStatic() 會被呼叫。

__callStatic 它的工作方式類似於 __call() 魔術方法,__callStatic() 是為了處理靜態方法呼叫,PHP5.3.0以上版本有效。
PHP 確實加強了對 __callStatic() 方法的定義;它必須是公共的,並且必須被宣告為靜態的。同樣,__call() 魔術方法必須被定義為公共的,所有其他魔術方法都必須如此。

4、__autoload

__autoload 函式,它會在試圖使用尚未被定義的類時自動呼叫。通過呼叫此函式,指令碼引擎在 PHP 出錯失敗前有了最後一個機會載入所需的類。
注意:在 __autoload 函式中丟擲的異常不能被 catch 語句塊捕獲並導致致命錯誤。

5、__construct、__destruct

__construct 構造方法,當一個物件建立時呼叫此方法。
使用此方法的好處是:可以使構造方法有一個獨一無二的名稱,無論它所在的類的名稱是什麼。這樣你在改變類的名稱時,就不需要改變構造方法的名稱。
__destruct 析構方法,PHP將在物件被銷燬前(即從記憶體中清除前)呼叫這個方法。
預設情況下,PHP僅僅釋放物件屬性所佔用的記憶體並銷燬物件相關的資源。
解構函式允許你在使用一個物件之後執行任意程式碼來清除記憶體。
當PHP決定你的指令碼不再與物件相關時,解構函式將被呼叫。
在一個函式的名稱空間內,這會發生在函式return的時候。
對於全域性變數,這發生於指令碼結束的時候。如果你想明確地銷燬一個物件,你可以給指向該物件的變數分配任何其它值,通常將變數賦值勤為NULL或者呼叫unset。

6、__clone

PHP5中的物件賦值是使用的引用賦值,如果想複製一個物件則需要使用clone方法,在呼叫此方法是物件會自動呼叫__clone魔術方法。
如果在物件複製需要執行某些初始化操作,可以在__clone方法實現。

7、__toString

複製程式碼 程式碼如下:
public string __toString ( void )

__toString方法在將一個物件轉化成字串時自動呼叫,比如使用echo列印物件時。 應該顯示些什麼。此方法必須返回一個字串,否則將發出一條 E_RECOVERABLE_ERROR 級別的致命錯誤。

複製程式碼 程式碼如下:
$myObject = new myClass();
echo $myObject;
// Will look for a magic method echo
$myObject->__toString();

注意:不能在 __toString() 方法中丟擲異常。這麼做會導致致命錯誤。

需要指出的是在 PHP 5.2.0 之前,__toString() 方法只有在直接使用於 echo 或 print 時才能生效。PHP 5.2.0 之後,則可以在任何字串環境生效(例如通過 printf(),使用 %s 修飾符),但不能用於非字串環境(如使用 %d 修飾符)。自 PHP 5.2.0 起,如果將一個未定義 __toString() 方法的物件轉換為字串,會產生 E_RECOVERABLE_ERROR 級別的錯誤。

8、__sleep() 和 __wakeup()

複製程式碼 程式碼如下:
public array __sleep ( void )
void __wakeup ( void )

serialize() 函式會檢查類中是否存在一個魔術方法 __sleep()。如果存在,該方法會先被呼叫,然後才執行序列化操作。此功能可以用於清理物件,並返回一個包含物件中所有應被序列化的變數名稱的陣列。如果該方法未返回任何內容,則 NULL 被序列化,併產生一個 E_NOTICE 級別的錯誤。

注意:

__sleep() 不能返回父類的私有成員的名字,這樣做會產生一個 E_NOTICE 級別的錯誤。可以用 Serializable 介面來替代。
__sleep() 方法常用於提交未提交的資料,或類似的清理操作。同時,如果有一些很大的物件,但不需要全部儲存,這個功能就很好用。
與之相反, unserialize() 會檢查是否存在一個 __wakeup() 方法。如果存在,則會先呼叫 __wakeup 方法,預先準備物件需要的資源。
__wakeup() 經常用在反序列化操作中,例如重新建立資料庫連線,或執行其它初始化操作。

9、__invoke()

複製程式碼 程式碼如下:mixed __invoke ([ $… ] )

當嘗試以呼叫函式的方式呼叫一個物件時,__invoke() 方法會被自動呼叫。

10、__set_state()

複製程式碼 程式碼如下:static object __set_state ( array $properties )

自 PHP 5.1.0 起當呼叫 var_export() 匯出類時,此靜態方法會被呼叫。
本方法的唯一引數是一個陣列,其中包含按 array(‘property’ => value, …) 格式排列的類屬性。