php的魔術方法

2009-10-04 20:31

從php5以後的版本,類就可以使用魔術方法了。php規定以兩個下劃線(__)開頭的方法都保留爲魔術方法,所以建議大家函數名最好不用__開頭,除非是爲了重載已有的魔術方法。

目前php已有的魔術方法有__construct,__destruct,__call,__get,__set,__isset,__unset,__sleep,__wakeup,__toString,__set_state 和 __clone

__construct和__destruct是類的構造函數和析構函數,這個大家經常會用到,相信大家都很熟悉,這裏就不多說了。

__sleep和__wakeup是序列化類的時候調用的。當序列化對象時,php將試圖在序列動作之前調用該對象的成員函數__sleep(),當使用unserialize() 恢復對象時, 將調用__wakeup()。

 

__toString是對象被轉爲string時調用的

 

詳情參考http://mydraft.cn/posts/66

1.__construct()

當實例化一個對象的時候,這個對象的這個方法首先被調用。

class Test
{
function __construct()

{
echo "before";
}
}


$t = new Test();

輸出是:

start
我們知道php5對象模型 和類名相同的函數是類的構造函數,那麼如果我們同時定義構造函數和__construct()方法的話,php5會默認調用構造函數而不會調用__construct()函數,所以__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;

程序輸出:
我被克隆了!

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

 

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