對(duì)象傳遞:一種說(shuō)法是“PHP對(duì)象是通過(guò)引用傳遞的”,更準(zhǔn)確的說(shuō)法是別名(標(biāo)識(shí)符)傳遞,即它們都保存著同一個(gè)標(biāo)識(shí)符(ID)的拷貝,這個(gè)標(biāo)識(shí)符指向同一個(gè)對(duì)象的真正內(nèi)容。
<?php class A { public $foo = 1; } $a = new A; $b = $a; // $a ,$b都是同一個(gè)標(biāo)識(shí)符的拷貝 // ($a) = ($b) = <id> $b->foo = 2; echo $a->foo."\n";//2 $c = new A; $d = &$c; // $c ,$d是引用 // ($c,$d) = <id> $d->foo = 2; echo $c->foo."\n";//2 $e = new A; function foo($obj) { // ($obj) = ($e) = <id> $obj->foo = 2; } foo($e); echo $e->foo."\n";//2
•對(duì)象復(fù)制:對(duì)象復(fù)制可以通過(guò) clone 關(guān)鍵字來(lái)完成,如果原對(duì)象定義了 __clone() 方法,則新對(duì)象中的 __clone() 方法將在復(fù)制完后被調(diào)用,__clone() 方法可用于修改復(fù)制對(duì)象屬性的值。當(dāng)對(duì)象被復(fù)制后,會(huì)對(duì)對(duì)象的所有屬性執(zhí)行一個(gè)淺復(fù)制(shallow copy),但所有的引用屬性仍然會(huì)是一個(gè)指向原來(lái)的變量的引用。
<?php class SubObject { static $instances = 0; public $instance; public function __construct() { $this->instance = ++self::$instances; } public function __clone() { $this->instance = ++self::$instances; } } class MyCloneable { public $object1; public $object2; function __clone() { // 強(qiáng)制復(fù)制一份this->object, 否則仍然指向同一個(gè)對(duì)象 $this->object1 = clone $this->object1; } function cloneTest() { echo 'cloneTest'; } } $obj = new MyCloneable(); $obj->object1 = new SubObject(); $obj->object2 = new SubObject(); $obj2 = clone $obj; print("Original Object:\n"); print_r($obj); print("Cloned Object:\n"); print_r($obj2); echo $obj2->cloneTest().":\n"; echo (new Reflectionclass($obj2));
上例輸出結(jié)果:
Original Object: MyCloneable Object ( [object1] => SubObject Object ( [instance] => 1 ) [object2] => SubObject Object ( [instance] => 2 ) ) Cloned Object: MyCloneable Object ( [object1] => SubObject Object ( [instance] => 3 ) [object2] => SubObject Object ( [instance] => 2 ) ) cloneTest: Class [ <user> class MyCloneable ] { @@ /public/t.php 18-33 - Constants [0] { } - Static properties [0] { } - Static methods [0] { } - Properties [2] { Property [ <default> public $object1 ] Property [ <default> public $object2 ] } - Methods [2] { Method [ <user> public method __clone ] { @@ /public/t.php 23 - 27 } Method [ <user> public method cloneTest ] { @@ /public/t.php 29 - 32 } } }
•對(duì)象遍歷: foreach只能遍歷對(duì)象的可見(jiàn)屬性,無(wú)法遍歷其方法,實(shí)現(xiàn)起來(lái)比較容易;另外,也可通過(guò)實(shí)現(xiàn)Iterator接口或IteratorAggregate接口的方法遍歷對(duì)象屬性。
•類(lèi)型約束: PHP作為一種弱類(lèi)型語(yǔ)言,類(lèi)型約束可以讓編程更加規(guī)范,也少出些差錯(cuò);類(lèi)型約束不只能用在對(duì)象定義中,也能用在函數(shù)定義中。類(lèi)型約束可指定對(duì)象、接口、array、callable(閉包c(diǎn)allback),類(lèi)型約束用來(lái)保證實(shí)際數(shù)據(jù)類(lèi)型與原型定義一致,不一致則拋出一個(gè)可捕獲的致命錯(cuò)誤;不過(guò)如果定義了默認(rèn)值為NULL,那么實(shí)參可以是NULL;類(lèi)型約束不能用于標(biāo)量類(lèi)型如 int 或 string,Traits 也不允許。
•對(duì)象序列化與還原:函數(shù)serialize()可將打成包含字節(jié)流的字符串便于存儲(chǔ)對(duì)象,函數(shù)unserialize()能夠還原字符串為對(duì)象。但有一個(gè)前提是,無(wú)論序列化還是反序列化,對(duì)象的類(lèi)定義已經(jīng)完成,即需要先導(dǎo)入類(lèi)(文件)。
•重載:PHP的重載包括屬性和方法,更像一個(gè)套用說(shuō)法,不支持常見(jiàn)的重載語(yǔ)法規(guī)范,具有不可預(yù)見(jiàn)性,影響范圍更寬泛,就是利用魔術(shù)方法(magic methods)來(lái)調(diào)用當(dāng)前環(huán)境下未定義或不可見(jiàn)的類(lèi)屬性或方法。所有重載方法都必須被聲明為 public(這一條應(yīng)該比較好理解,別人可能因不可見(jiàn)才需要你,那你自己必須可見(jiàn)才行),參數(shù)也不能通過(guò)引用傳遞(重載方法具有不可預(yù)見(jiàn)性,估計(jì)出于安全方面的考慮吧,防止變量被隨意引用)。在除 isset() 外的其它語(yǔ)言結(jié)構(gòu)中無(wú)法使用重載的屬性,這意味著當(dāng)對(duì)一個(gè)重載的屬性使用 empty() 時(shí),重載魔術(shù)方法將不會(huì)被調(diào)用; 為避開(kāi)此限制,必須將重載屬性賦值到本地變量再使用 empty(),可見(jiàn)重載屬性是介于合法屬性與非法屬性之間的存在。
[屬性重載]:這些方法不能被聲明為 static,在靜態(tài)方法中,這些魔術(shù)方法將不會(huì)被調(diào)用
public void __set ( string $name , mixed $value )
在給不可訪問(wèn)屬性賦值時(shí),__set() 會(huì)被調(diào)用
public mixed __get ( string $name )
讀取不可訪問(wèn)屬性的值時(shí),__get() 會(huì)被調(diào)用
public bool __isset ( string $name )
當(dāng)對(duì)不可訪問(wèn)屬性調(diào)用 isset() 或 empty() 時(shí),__isset() 會(huì)被調(diào)用
public void __unset ( string $name )
當(dāng)對(duì)不可訪問(wèn)屬性調(diào)用 unset() 時(shí),__unset() 會(huì)被調(diào)用