在多數(shù)情況下,我們并不需要完全復制一個對象來獲得其中屬性。但有一個情況下確實需要:如果你有一個 GTK 窗口對象,該對象持有窗口相關(guān)的資源。你可能會想復制一個新的窗口,保持所有屬性與原來的窗口相同,但必須是一個新的對象(因為如果不是新的對象,那么一個窗口中的改變就會影響到另一個窗口)。還有一種情況:如果對象 A 中保存著對象 B 的引用,當你復制對象 A 時,你想其中使用的對象不再是對象 B 而是 B 的一個副本,那么你必須得到對象 A 的一個副本。
對象復制可以通過 clone
關(guān)鍵字來完成(如果可能,這將調(diào)用對象的
__clone() 方法)。
$copy_of_object = clone $object;
當對象被復制后,PHP 會對對象的所有屬性執(zhí)行一個淺復制(shallow copy)。所有的引用屬性 仍然會是一個指向原來的變量的引用。
當復制完成時,如果定義了 __clone() 方法,則新創(chuàng)建的對象(復制生成的對象)中的 __clone() 方法會被調(diào)用,可用于修改屬性的值(如果有必要的話)。
示例 #1 復制一個對象
<?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()
{
// 強制復制一份this->object, 否則仍然指向同一個對象
$this->object1 = clone $this->object1;
}
}
$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);
?>
以上例程會輸出:
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 ) )
可以在單個表達式中訪問剛剛克隆的對象的成員:
示例 #2 訪問新克隆的對象的成員
<?php
$dateTime = new DateTime();
echo (clone $dateTime)->format('Y');
?>
以上例程的輸出類似于:
2016