PHP 魔術常量、函數詳解

魔術常量

__LINE__ 文件中的當前行號。 

__FILE__ 文件的完整路徑和文件名。如果用在被包含文件中,則返回被包含的文件名。自 PHP 4.0.2 起,__FILE__ 總是包含一個絕對路徑(如果是符號連接,則是解析後的絕對路徑),而在此之前的版本有時會包含一個相對路徑。 

__DIR__ 文件所在的目錄。如果用在被包括文件中,則返回被包括的文件所在的目錄。它等價於 dirname(__FILE__)。除非是根目錄,否則目錄中名不包括末尾的斜槓。(PHP 5.3.0中新增) = 

__FUNCTION__ 函數名稱(PHP 4.3.0 新加)。自 PHP 5 起本常量返回該函數被定義時的名字(區分大小寫)。在 PHP 4 中該值總是小寫字母的。 

__CLASS__ 類的名稱(PHP 4.3.0 新加)。自 PHP 5 起本常量返回該類被定義時的名字(區分大小寫)。在 PHP 4 中該值總是小寫字母的。 

__METHOD__ 類的方法名(PHP 5.0.0 新加)。返回該方法被定義時的名字(區分大小寫)。 

__NAMESPACE__ 當前命名空間的名稱(大小寫敏感)。這個常量是在編譯時定義的(PHP 5.3.0 新增)

 

 

魔術方法:

__construct__destruct (參看 構造方法和析構方法), __call__callStatic__get__set__isset__unset (參看 重載), __sleep__wakeup__toString__set_state 和 __clone等方法在PHP中被稱爲“魔術方法”(Magic methods)

1、__get、__set
這兩個方法是爲在類和他們的父類中沒有聲明的屬性而設計的
__get( $property ) 當調用一個未定義的屬性時訪問此方法
__set( $property, $value ) 給一個未定義的屬性賦值時調用
這裏的沒有聲明包括當使用對象調用時,訪問控制爲proteced,private的屬性(即沒有權限訪問的屬性)

  1. <?php  
  2. /* 當試圖調用不存在或者不可見的屬性時,調用__set(),必須要接受兩個參數,屬性名和屬性值*/  
  3. class myShop   
  4. {  
  5.     private $p = array();  
  6.     function __set($name,$value)  
  7.     {  
  8.         $this->p[$name] = $value;  
  9.     }  
  10.     function __get($name)  
  11.     {  
  12.         return array_key_exists($name,$this->p) ? $this->p[$name] : null;  
  13.     }  
  14. }  
  15. $shop = new myShop();  
  16. $shop->feiker=2;       //訪問不存在的屬性,調用__set();  
  17. $shop->pear = 5;  
  18. $shop->pear ++;  //先調用__get(),因一開始只有名字,之後調用__set();  
  19. echo $shop->feiker;  
  20. echo $shop->pear;  
  21. /*===============*/  
  22. class Test {  
  23.     public function __set($name,$value){  
  24.         $this->$name = $value;  
  25.     }  
  26. }  
  27. $test = new Test();  
  28. var_dump($test);  
  29. echo $test->xixi;  
  30. echo '<hr/>';  
  31. var_dump($test);  

 

2、__isset、__unset

外部調用isset(),unset()函數時,對於調用這兩個函數。

當類的外部,在判斷對象中是否存在屬性時,調用__set()對象中方法。默認只檢測可見性屬性,除非在類中添加__set()方法,返回對應屬性。

同理unset();

  1. <?php  
  2. //__unset()和__isset()  
  3. class person  
  4. {  
  5.     private $name,$sex,$age;//私有屬性  
  6.     function __get($property_name)  
  7.     {//獲取私有屬性的名稱  
  8.         if ( isset( $property_name ) ) echo $this->$property_name;  
  9.     }  
  10.     function __set($property_name,$value)  
  11.     {  
  12.         $this->$property_name = $value;  
  13.         echo "名稱爲:{$property_name}值爲:{$value}<br>";  
  14.     }  
  15.     function __isset($ff)  
  16.     {//__isset()  
  17.         echo "isset()函數測定私有成員屬性是,自動調用<br>";  
  18.         return isset($this->$ff);  
  19.     }  
  20.     function __unset($kk)  
  21.     {//__unset()  
  22.         echo "unset()是當在類的外部調用是,自動調用方法來刪除私有成員變量<br>";  
  23.         unset($this->$kk);  
  24.     }  
  25.     function a()  
  26.     {  
  27.         echo 'bb'.$this->name;  
  28.     }  
  29. }  
  30. $obj = new person();  
  31. $obj->name = "飛客";  
  32. //isset()測定私有成員變量  
  33. isset($obj->name);  
  34. //自動調用刪除私有成員變量  
  35. unset($obj->name);  
  36. echo $obj->a();  
  37. //echo 'aaa'.$obj->name.'bbb';        //result : 飛客aaabbb  
  38. ?>  

 

3、__call
__call( $method, $arg_array ) 當調用一個未定義的方法是調用此訪求
這裏的未定義的方法包括沒有權限訪問的方法

  1. <?php  
  2. /*當試圖調用類中一個不存在或者不可用的方法時,會執行該類中的__call()__call()必須接受兩個參數,第一個參數存放方法名稱,第二個參數存放不存在的方法的參數(此參數會放在與該參數同名的數組中)*/  
  3. class callclass  
  4. {  
  5.     function __call($method_name,$p)  
  6.     {  
  7.         echo "使用__call嘗試調用一個不存在/不可用的成員方法<br>";  
  8.         echo $method_name;  
  9.         echo "<pre>";  
  10.         print_r($p);  
  11.         echo "</pre>";  
  12.     }  
  13. }  
  14. $obj = new callclass();  
  15. $obj->method(1,2,"Hello","HP")  
  16. ?>  
 

 

4、__autoload
__autoload 函數,它會在試圖使用尚未被定義的類時自動調用。通過調用此函數,腳本引擎在 PHP 出錯失敗前有了最後一個機會加載所需的類。
注意: 在 __autoload 函數中拋出的異常不能被 catch 語句塊捕獲並導致致命錯誤。

  1. <?php  
  2. function __autoload($class_name) {  
  3.     require_once $class_name . '.php';  
  4. }  
 

 

5、__construct、__destruct
__construct 構造方法,當一個對象創建時調用此方法,使用此方法的好處是:可以使構造方法有一個獨一無二的名稱,無論它所在的類的名稱是什麼.這樣你在改變類的名稱時,就不需要改變構造方法的名稱
__destruct 析構方法,PHP將在對象被銷燬前(即從內存中清除前)調用這個方法
默認情況下,PHP僅僅釋放對象屬性所佔用的內存並銷燬對象相關的資源.
析構函數允許你在使用一個對象之後執行任意代碼來清除內存.
當PHP決定你的腳本不再與對象相關時,析構函數將被調用.
在一個函數的命名空間內,這會發生在函數return的時候.
對於全局變量,這發生於腳本結束的時候.如果你想明確地銷燬一個對象,你可以給指向該對象的變量分配任何其它值.通常將變量賦值勤爲NULL或者調用unset.

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

7、__toString 
__toString方法在將一個對象轉化成字符串時自動調用,比如使用echo打印對象時
如果類沒有實現此方法,則無法通過echo打印對象,否則會顯示:Catchable fatal error: Object of class test could not be converted to string in
此方法必須返回一個字符串

在PHP 5.2.0之前,__toString方法只有結合使用echo() 或 print()時 才能生效。PHP 5.2.0之後,則可以在任何字符串環境生效(例如通過printf(),使用%s修飾符),但 不能用於非字符串環境(如使用%d修飾符)。從PHP 5.2.0,如果將一個未定義__toString方法的對象 轉換爲字符串,會報出一個E_RECOVERABLE_ERROR錯誤。

  1. // Declare a simple class  
  2. class TestClass  
  3. {  
  4.     public $foo;  
  5.     public function __construct($foo) {  
  6.         $this->foo = $foo;  
  7.     }  
  8.     public function __toString() {  
  9.         return $this->foo;  
  10.     }  
  11. }  
  12. $class = new TestClass('Hello');  
  13. echo $class;  
 


8、__sleep、__wakeup
__sleep 串行化的時候用
__wakeup 反串行化的時候調用
serialize() 檢查類中是否有魔術名稱 __sleep 的函數。如果這樣,該函數將在任何序列化之前運行。它可以清除對象並應該返回一個包含有該對象中應被序列化的所有變量名的數組。
使用 __sleep 的目的是關閉對象可能具有的任何數據庫連接,提交等待中的數據或進行類似的清除任務。此外,如果有非常大的對象而並不需要完全儲存下來時此函數也很有用。
相反地,unserialize() 檢查具有魔術名稱 __wakeup 的函數的存在。如果存在,此函數可以重建對象可能具有的任何資源。
使用 __wakeup 的目的是重建在序列化中可能丟失的任何數據庫連接以及處理其它重新初始化的任務。

9、__set_state
當調用var_export()時,這個靜態 方法會被調用(自PHP 5.1.0起有效)。
本方法的唯一參數是一個數組,其中包含按array(’property’ => value, …)格式排列的類屬性。

10、__invoke
當嘗試以調用函數的方式調用一個對象時,__invoke 方法會被自動調用。
PHP5.3.0以上版本有效


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


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章