2009年11月11日 星期三

[PHP] PHP5 的 Class 函數

PHP4中已經有了重載的語法來建立對於外部對象模型的映射,就像Java和COM那樣. PHP5帶來了強大的面向對象重載,允許程序員建立自定義的行為來訪問屬性和調用方法,php5加入了如下的內部特徵

__construct(); 初始化--構造函數

__destruct(); 卸載--析構函數

__get(); __get方法可以用來捕獲一個對象中不存在的變量和方法

__set(); __set方法可以用來捕獲和按參數修改一個對象中不存在的變量和方法

__call(); 調用不存在的類的函數的時候得處理方法

__clone(); copy對象用clone $obj;

__sleep(); 串行化的時候用

__wakeup(); 反串行化的時候用

重載可以通過__get, __set, and __call幾個特殊方法來進行. 當Zend引擎試圖訪問一個成員並沒有找到時,PHP將會調用這些方法.

在例6.14中,__get和__set代替所有對屬性變量數組的訪問. 如果必要,你可以實現任何類型你想要的過濾. 例如,腳本可以禁止設置屬性值, 在開始時用一定的前綴或包含一定類型的值.

__call方法說明了你如何調用未經定義的方法. 你調用未定義方法時,方法名和方法接收的參數將會傳給__call方法, PHP傳遞__call的值返回給未定義的方法.

--------------------------------------------------------------------------------


<?php
class foo {
function __set($name,$val) {
print("
Hello, you tried to put $val in $name");
}
function __get($name) {
print("
Hey you asked for $name");
}
}
$x = new foo();
$x->bar = 3;//注意$bar不存在
print($x->winky_winky);
?>


--------------------------------------------------------------------------------


<?php
class foo {
function __call($name,$arguments) {
print("Did you call me? I'm $name!");
}
} $x = new foo();
$x->doStuff(2);
$x->fancy_stuff('a string');
?>

這個特殊的方法可以被用來實現「過載(overloading)」的動作,這樣你就可以檢查你的參數並且通過調用一個私有的方法來傳遞參數。

使用 __call 實現「過載」動作

--------------------------------------------------------------------------------

<?php
class Magic {
function __call($name,$arguments) {
if($name=='foo') {
print_r($arguments);
if(is_int($arguments[0])) $this->foo_for_int($arguments[0]);
if(is_string($arguments[0])) $this->foo_for_string($arguments[0]);
}
} private function foo_for_int($x) {
print("
oh an int!");
} private function foo_for_string($x) {
print("
oh a string!");
}
} $x = new Magic();
$x->foo(3);
$x->foo("3");
?>

對一個對象的拷貝通過調用對象的__clone()方法完成

--------------------------------------------------------------------------------

<?php
$copy_of_object =clone $obj;
?>

--------------------------------------------------------------------------------

<?php
class MyCloneable {
static $id = 0;

function MyCloneable() {
$this->id = self::$id+1; //注意這裡如果寫self::$id++;將不被充許
}

function __clone() {
$this->address = "New York";
$this->id = self::$id+1;
}
}

$obj = new MyCloneable();

$obj->name = "Hello";
$obj->address = "Tel-Aviv";

print $obj->id . "
";

$obj =clone $obj;

print $obj->id . "
";
print $obj->name . "
";
print $obj->address . "
";
?>

串行化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 serialization

--------------------------------------------------------------------------------

<?php

class User
{
public $name;
public $id;

function __construct()
{
//give user a unique ID 賦予一個不同的ID
$this->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);
?>

【下列文章您可能也有興趣】

沒有留言: