Php魔術函數學習與應用 __construct() __destruct() __get()等
(1)初識魔術方法
Php5.0發布以來為我們提供了很多面向對象?的特性,尤其是為我們提供了好多易用的魔術方法,這些魔術方法可以讓我們簡化我們的編碼,更好的設計我們的系統。今天我們就來認識下php5.0給我們提供的魔術方法。
1,__construct() 當執行個體化一個對象的時候,這個對象的這個方法首先被調用。
class Test{function __construct(){echo "before";}}$t = new Test();
?
輸出是:
start
?
//輸出end
我們知道php5物件模型?和類名相同?的函數?是類的構造函數?,那麼如果我們同時定義構造函數?和__construct()方法的話,php5會預設調用構造函數?而不會調用同類名函數?,所以__construct()作為類的預設的構造函數?
2,__destruct() 當刪除一個對象或對象操作終止的時候,調用該方法。
class Test{function __destruct(){echo "end";}}$t = new Test();
?
將會輸出
end
我們就可以在對象操作結束的時候進行釋放資源之類的操作
3,__get() 當試圖讀取一個並不存在的屬性的時候被調用。
如果試圖讀取一個對象並不存在的屬性的時候,PHP就會給出錯誤資訊。如果在類裡添加__get方法,並且我們可以用這個函數?實作類別似java中反射的各種操作。
class Test{public function __get($key){echo $key . " 不存在";}}$t = new Test();echo $t->name;
?
就會輸出:
name 不存在
4,__set() 當試圖向一個並不存在的屬性寫入值的時候被調用。
class Test{public function __set($key,$value){echo '對'.$key . "附值".$value;}}$t = new Test();$t->name = "aninggo";
?
就會輸出:
對 name 附值 aninggo
5,__call() 當試圖調用一個對象並不存在的方法時,調用該方法。
class Test{public function __call($Key, $Args){echo "您要調用的 {$Key} 方法不存在。你傳入的參數是:" . print_r($Args, true);}}$t = new Test();$t->getName(aning,go);
?
程式將會輸出:
您要調用的 getName 方法不存在。參數是:Array
(
[0] => aning
[1] => go
)
6,__toString() 當列印一個對象的時候被調用
這個方法類似於java的toString方法,當我們直接列印對象的時候回調用這個函數?
class Test{public function __toString(){return "列印 Test";}}$t = new Test();echo $t;
?
運行echo $t;的時候,就會調用$t->__toString();從而輸出
列印 Test
7,__clone() 當對象被複製時,被調用
class Test{public function __clone(){echo "我被複製了!";}}$t = new Test();$t1 = clone $t;
?
程式輸出:
我被複製了!
__sleep 和 __wakeup
序列化serialize可以把變數包括對象,轉化成連續bytes資料. 你可以將序列化後的變數存在一個檔案裡或在網路上傳輸. 然後再反序列化還原為原來的資料. 你在反序列化類的對象之前定義的類,PHP可以成功地儲存其對象的屬性和方法. 有時你可能需要一個對象在反序列化後立即執行. 為了這樣的目的,PHP會自動尋找__sleep和__wakeup方法.
當一個對象被序列化,PHP會調用__sleep方法(如果存在的話). 在反序列化一個對象後,PHP 會調用__wakeup方法. 這兩個方法都不接受參數. __sleep方法必須返回一個數組,包含需要序列化的屬性. PHP會拋棄其它屬性的值. 如果沒有__sleep方法,PHP將儲存所有屬性.
例子6.16顯示了如何用__sleep和__wakeup方法來序列化一個對象. Id屬性是一個不打算保留在對象中的臨時屬性. __sleep方法保證在序列化的對象中不包含id屬性. 當反序列化一個User對象,__wakeup方法建立id屬性的新值. 這個例子被設計成自我保持. 在實際開發中,你可能發現包含資源(像或資料流)的對象需要這些方法
Object serializationCODE: [Copy to clipboard]--------------------------------------------id = uniqid(); } function __sleep() { //do not serialize this->id 不序列化id return(array("name")); } function __wakeup() { //give user a unique ID $this->id = uniqid(); } } //create object 建立一個對象 $u = new User; $u->name = "Leon"; //serialize it 序列化 注意不序列化id屬性,id的值被拋棄 $s = serialize($u); //unserialize it 反序列化 id被重新賦值 $u2 = unserialize($s); //$u and $u2 have different IDs $u和$u2有不同的ID print_r($u); print_r($u2); ?>?
__set_state and __invoke
測試代碼如下:
$v) { $obj->$k = $v; } return $obj; }}$a = new A;$a->name = 'cluries';$a->sex = 'female';eval('$b = ' . var_export($a, true) . ';');print_r($b);?>??
程式輸出
object(A)#2 (2) { ["name"]=> string(7) "cluries" ["sex"]=> string(6) "female"}
得出以下結論,__set_state作用是用來複製一個對象,並且可以在__set_state中定義在複製對象的時候對複製得到的對象進行一些 改變。和__clone不同的是__set_state可以接受參數,__set_state使用起來更加強大!雖然個人覺得這個東西不是很好用= =!
然後再說下__invoke:
手冊上有個非常顯眼的:Note: This feature is available since PHP 5.3.0.?
The __invoke method is called when a script tries to call an object as a function.
__invoke方法將會在代碼試圖把對象當作函數來使用時候調用?有點稀奇,這個功能有什麼用處呢?
然後看下提供的例子:
?
程式輸出:
int(5)bool(true)
還真是把對象當函數使用…
__autoload
PHP5中有一方法: __autoload() , 簡單的說就是類的自動載入;
| 當你嘗試使用一個PHP沒有組織到的類, 它會尋找一個__autoload的全域函數. 如果存在這個函數,PHP會用一個參數來調用它,參數即類的名稱。 |
那麼簡單測試一下。
首先建一個名為”Test_autoload.php”的檔案:
< ? php/*** 測試__autoload方法**/class Test_autoload {public function __construct () {echo " Test_autoload. " ;}}?>?
注意類名哦?, 然後隨便建個檔案重寫 __autoload() 方法,這裡假設是”test.php”;
< ? php/*** 重寫 __autoload方法*/function __autoload ( $class ) {include $class . ' .php ' ;}$test = new Test_autoload () ;unset ( $test ) ;?>?
最後結果為:Test_autoload.
------------------------------------------------
8.順便介紹下php5中提供的幾個非常COOl的實驗性函數?
(1)。runkit_method_rename
??? 這個函數?可以動態改變我們所調用的函數?的名字?。
class Test{function foo() { return "foo! "; }}runkit_method_rename( 'Test', //類名 'foo',//實際調用的函數 'bar'//顯示調用的函數 );echo Test::bar();
?
程式將輸出
foo!
(2) runkit_method_add
這個函數?可以動態向類中添加函數?
class Test{function foo() { return "foo! "; }}runkit_method_add( Test, //類名 'add', //新函數 名 '$num1, $num2',//傳入參數 'return $num1 + $num2;',//執行的代碼 RUNKIT_ACC_PUBLIC);// 調用echo $e->add(12, 4);
?
(3)runkit_method_copy
可以把A類中的函數?拷貝到類B中並對函數?重新命名
class Foo { function example() { return "foo! "; }}class Bar { //空類}//執行拷貝runkit_method_copy('Bar', 'baz', 'Foo', 'example');//執行拷貝後的函數 echo Bar::baz();
?
(4) runkit_method_redefine
動態修改函數?的傳回值
這個函數?可以讓我們輕鬆的實現對類的MOCK測試!是不是很COOL呢
class Example { function foo() { return "foo! "; }}//建立一個測試對象$e = new Example();// 在測試對象之前輸出echo "Before: " . $e->foo();// 修改傳回值runkit_method_redefine( 'Example', 'foo', '', 'return "bar! ";', RUNKIT_ACC_PUBLIC);// 執行輸出echo "After: " . $e->foo();
?
(5)runkit_method_remove
這個函數?就很簡單了,看名字?就能看出來了,動態從類中移除函數?
class Test { function foo() { return "foo! "; } function bar() { return "bar! "; }}// 移除foo函數 runkit_method_remove( 'Test', 'foo');echo implode(' ', get_class_methods('Test'));
?
程式輸出
bar
?