PHP中魔術方法的定義是把以兩個底線__開頭的方法稱為魔術方法,這些魔術方法在PHP中的作用是非常重要的,下面我們就來看看這些魔術方法的執行個體。
魔術方法:
__construct(),類的建構函式__destruct(),類的解構函式__call(),在對象中調用一個不可存取方法時調用__callStatic(),用靜態方式中調用一個不可存取方法時調用__get(),獲得一個類的成員變數時調用__set(),設定一個類的成員變數時調用__isset(),當對不可訪問屬性調用isset()或empty()時調用__unset(),當對不可訪問屬性調用unset()時被調用。__sleep(),執行serialize()時,先會調用這個函數__wakeup(),執行unserialize()時,先會調用這個函數__toString(),類被當成字串時的回應方法__invoke(),調用函數的方式調用一個對象時的回應方法__set_state(),調用var_export()匯出類時,此靜態方法會被調用。__clone(),當對象複製完成時調用
__construct()和__destruct()
建構函式和解構函式應該不陌生,他們在對象建立和消亡時被調用。例如我們需要開啟一個檔案,在對象建立時開啟,對象消亡時關閉
<?phpclass FileRead{ protected $handle = NULL; function __construct(){ $this->handle = fopen(...); } function __destruct(){ fclose($this->handle); }}?>
這兩個方法在繼承時可以擴充,例如:
<?phpclass TmpFileRead extends FileRead{ function __construct(){ parent::__construct(); } function __destruct(){ parent::__destruct(); }}?>
__call()和__callStatic()
在對象中調用一個不可存取方法時會調用這兩個方法,後者為靜態方法。這兩個方法我們在可變方法(Variable functions)調用中可能會用到。
<?phpclass MethodTest{ public function __call ($name, $arguments) { echo "Calling object method '$name' ". implode(', ', $arguments). "\n"; } public static function __callStatic ($name, $arguments) { echo "Calling static method '$name' ". implode(', ', $arguments). "\n"; }} $obj = new MethodTest;$obj->runTest('in object context');MethodTest::runTest('in static context');?>
__get(),__set(),
__get屬性是當訪問對象中的屬性不存在或者非公有屬性的時候自動載入__get方法,參數只有一個name值,就是訪問對象中
屬性的名字。
__set是當給對象中一個屬性賦值的時候如果該屬性不存在或者非公有屬性的時候就會自動載入__set方法。參數有兩個,參數1是訪問對象中不可被調用屬性的名字,參數2是將要給賦值所傳的參數,可以是數組,也可以是字串
都具有公用可見度,非靜態,舉個小栗子方便理解:
<?php/** * 清晰的認識__get() __set() */class Example { //公有的屬性 public $public = 'pub' ; //受保護的 - 子類中該屬性可用 protected $protected = 'pro'; //私人的 - 只能此類使用此屬性 private $private = 'pri'; //當訪問對象中的屬性不存在或者非公有屬性的時候自動載入__get()方法 public function __get($name){ return '調用__get()方法:'.$name; } //當給對象的一個屬性賦值的時候如果該屬性不存在或者是非公有屬性則自動載入__set()方法 public function __set($name,$value){ echo "\nname:".$name.',value:'.$value."\n"; }}$example = new Example;echo '<pre>';echo $example->public."\n";echo $example->protected."\n";echo $example->private."\n";echo $example->other."\n";echo '<hr>';$example->public = 'lic'; //這個賦值成功所有沒有顯示$example->protected = 'tec';$example->private = 'vat';$example->other = 'er';echo '<br/>';
echo '列印 public 屬性:'.$example->public;
__isset()和__unset()
__isset()在對象中調用屬性或者非類中調用屬性使用isset()方法的時候如果沒有或者非公有屬性
就會自動執行isset()的方法
__unset()在對象中調用屬性或者非類中調用屬性使用unset()方法如果沒有或者非公有屬性就會
自動執行__unset()的調用,可以將不能調用的成員屬性刪除,如果沒有在類中加入此方法就不能刪除
對象中的任何私人成員,
一起舉個小栗子先:
<?php /** * 針對類中的魔術方法 __isset() 和 __unset() 的例子 */class Example { public $public; protected $protected; private $private; public function __construct(){ $this->public = 'pub'; $this->protected = 'pro'; $this->private = 'pri'; } public function __isset($var){ echo '這裡通過__isset()方法查看屬性名稱為 '.$var."\n"; } public function __unset($var){ echo '這裡通過__unset()方法要銷毀屬性名稱為 '.$var."\n"; }}$exa = new Example;echo '<pre>';var_dump(isset($exa->public));echo "\n";var_dump(isset($exa->protected));echo "\n";var_dump(isset($exa->private));echo "\n";var_dump(isset($exa->noVar));echo "\n";echo '<hr/>';unset($exa->public);var_dump($exa);echo "\n";unset($exa->protected);echo "\n";unset($exa->private);echo "\n";unset($exa->noVar);echo "\n";
__sleep()和__wakeup()
當我們在執行serialize()和unserialize()時,會先調用這兩個函數。例如我們在序列化一個對象時,這個對象有一個資料庫連結,想要在還原序列化中恢複連結狀態,則可以通過重構這兩個函數來實現連結的恢複。例子如下:
<?phpclass Connection{ protected $link; private $server, $username, $password, $db; public function __construct($server, $username, $password, $db) { $this->server = $server; $this->username = $username; $this->password = $password; $this->db = $db; $this->connect(); } private function connect() { $this->link = mysql_connect($this->server, $this->username, $this->password); mysql_select_db($this->db, $this->link); } public function __sleep() { return array('server', 'username', 'password', 'db'); } public function __wakeup() { $this->connect(); }}?>
__toString()
對象當成字串時的回應方法。例如使用echo $obj;來輸出一個對象
<?php// Declare a simple classclass TestClass{ public function __toString() { return 'this is a object'; }} $class = new TestClass();echo $class;?>
這個方法只能返回字串,而且不可以在這個方法中拋出異常,否則會出現致命錯誤。
__invoke()
調用函數的方式調用一個對象時的回應方法。如下
<?phpclass CallableClass{ function __invoke() { echo 'this is a object'; }}$obj = new CallableClass;var_dump(is_callable($obj));?>
__set_state()
調用var_export()匯出類時,此靜態方法會被調用。
<?phpclass A{ public $var1; public $var2; public static function __set_state ($an_array) { $obj = new A; $obj->var1 = $an_array['var1']; $obj->var2 = $an_array['var2']; return $obj; }} $a = new A;$a->var1 = 5;$a->var2 = 'foo';var_dump(var_export($a));?>
__clone()
當對象複製完成時調用。例如在設計模式詳解及PHP實現:單例模式一文中提到的單例模式實現方式,利用這個函數來防止對象被複製。
<?phppublic class Singleton { private static $_instance = NULL; // 私人構造方法 private function __construct() {} public static function getInstance() { if (is_null(self::$_instance)) { self::$_instance = new Singleton(); } return self::$_instance; } // 防止複製執行個體 public function __clone(){ die('Clone is not allowed.' . E_USER_ERROR); }}?>
魔術常量(Magic constants)
PHP中的常量大部分都是不變的,但是有8個常量會隨著他們所在代碼位置的變化而變化,這8個常量被稱為魔術常量。
__LINE__,檔案中的當前行號
__FILE__,檔案的完整路徑和檔案名稱
__DIR__,檔案所在的目錄
__FUNCTION__,函數名稱
__CLASS__,類的名稱
__TRAIT__,Trait的名字
__METHOD__,類的方法名
__NAMESPACE__,當前命名空間的名稱
這些魔術常量常常被用於獲得當前環境資訊或者記錄日誌。