Php幻術函數學習與應用 _construct() _destruct() _get()等

來源:互聯網
上載者:User
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

?

  • 聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

    如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

    A Free Trial That Lets You Build Big!

    Start building with 50+ products and up to 12 months usage for Elastic Compute Service

    • Sales Support

      1 on 1 presale consultation

    • After-Sales Support

      24/7 Technical Support 6 Free Tickets per Quarter Faster Response

    • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.