PHP - 物件克隆



像 “$obj1 = $obj2” 這樣的 PHP 語句僅僅建立了對記憶體中同一物件的另一個引用。因此,屬性的更改會反映在原始物件和副本物件中。PHP 中的 **clone** 關鍵字建立物件的淺複製。

$obj2 = $obj1

原始物件的更改不會反映在淺複製中。

示例

請看以下示例:

<?php
   class foo {
      var $var1 = 'Hello';
   }
   $x = new foo();
   $y = $x;		# reference copy
   echo $x->var1 . " " . $y->var1 . PHP_EOL;

   $x->var1 = "Hello World";
   echo $x->var1 . " " . $y->var1 . PHP_EOL;
?>

它將產生以下 **輸出**:

Hello Hello
Hello World Hello World

在第一種情況下,**$y** 只是 **$x** 的引用副本。因此,**var1** 屬性的任何更改都會反映在兩者中。

但是,如果我們將 **$y** 宣告為 **$x** 的克隆,則原始物件的任何更改都不會反映在其淺複製中。

示例

請看以下示例:

<?php
   class foo {
      var $var1 = 'Hello World';
   }

   $x = new foo();

   # shallow copy
   $y = clone $x;
   echo $x->var1 . " " . $y->var1 . PHP_EOL;

   $x->var1 = "Hello PHP";
   echo $x->var1 . " " . $y->var1 . PHP_EOL;
?>

它將產生以下 **輸出**:

Hello World Hello World
Hello PHP Hello World

示例

在以下程式碼中,**myclass** 的一個屬性是 address 類的物件。透過賦值複製 myclass 的物件。其嵌入的 address 物件的值的任何更改都將反映在兩個物件中,但 name 屬性的更改不會影響克隆的物件。

<?php
   class address {
      var $city="Nanded";
      var $pin="431601";
      function setaddr($arg1, $arg2) {
         $this->city=$arg1;
         $this->pin=$arg2;
      }
   }
   class myclass {
      var $name="Raja";
      var $obj;
      function setname($arg) {
         $this->name=$arg;
      }
   }

   $obj1=new myclass();
   $obj1->obj=new address();
   echo "original object\n";
   print_r($obj1);
   echo "\n";

   $obj2=$obj1;		# reference copy
   $obj1->setname("Ravi");
   $obj1->obj->setaddr("Mumbai", "400001");
   echo "after change: Original object\n";
   print_r($obj1);
   echo "\nCopied object\n";
   print_r($obj2);
?>

它將產生以下 **輸出**:

original object
myclass Object
(
   [name] => Raja
   [obj] => address Object
   (
      [city] => Nanded
      [pin] => 431601
   )
)

after change: Original object
myclass Object
(
   [name] => Ravi
   [obj] => address Object
   (
      [city] => Mumbai
      [pin] => 400001
   )
)

Copied object
myclass Object
(
   [name] => Ravi
   [obj] => address Object
   (
      [city] => Mumbai
      [pin] => 400001
   )
)

使用 "clone" 關鍵字

在淺複製中,原始物件的任何引用其他變數的屬性都將保持引用。clone 關鍵字不會複製被複制物件的包含物件。

現在我們建立 myclass 物件的克隆,以便 **$obj2** 是 **$obj1** 的克隆。我們將 **$obj1** 的 name 屬性從 **Raja** 更改為 **Ravi**,並修改嵌入的 address 物件。屬性更改將不會反映在其克隆中,但引用的 address 物件將被更改。

示例

請看以下示例:

<?php
   class address {
      var $city="Nanded";
      var $pin="431601";
      function setaddr($arg1, $arg2) {
         $this->city=$arg1;
         $this->pin=$arg2;
      }
   }
   class myclass {
      var $name="Raja";
      var $obj;
      function setname($arg) {
         $this->name=$arg;
      }
   }
   $obj1=new myclass();
   $obj1->obj=new address();
   echo "original object\n";
   print_r($obj1);
   echo "\n";

   $obj2=clone $obj1;		# clone copy
   $obj1->setname("Ravi");
   $obj1->obj->setaddr("Mumbai", "400001");
   echo "after change: Original object\n";
   print_r($obj1);
   echo "\nCopied object\n";
   print_r($obj2);
?>

它將產生以下 **輸出**:

original object
myclass Object
(
   [name] => Raja
   [obj] => address Object
   (
      [city] => Nanded
      [pin] => 431601
   )
)

after change: Original object
myclass Object
(
   [name] => Ravi
   [obj] => address Object
   (
      [city] => Mumbai
      [pin] => 400001
   )
)

Copied object
myclass Object
(
   [name] => Raja
   [obj] => address Object
   (
      [city] => Mumbai
      [pin] => 400001
   )
)

使用 __clone() 方法

clone 關鍵字建立物件的淺複製。當克隆物件時,PHP 將執行其所有屬性的淺複製。任何引用其他變數的屬性都將保持引用。因此,對原始物件所做的任何更改也將在克隆物件中出現。

如果您希望阻止複製的物件自動更新,我們需要使用 __clone() 方法建立物件的深複製。它是 PHP 中的一種魔術方法。

克隆完成後,如果定義了 __clone() 方法,則將呼叫新建立的物件的 __clone() 方法,以允許更改任何需要更改的必要屬性。

示例

在上面的示例中,我們有一個 myclass 的物件,它的一個屬性 $obj 持有對 address 類物件的引用。為了實現深複製,我們在 myclass 中覆蓋了 __clone() 魔術方法。

<?php
   class address {
      var $city="Nanded";
      var $pin="431601";
      function setaddr($arg1, $arg2) {
         $this->city=$arg1;
         $this->pin=$arg2;
      }
   }
   class myclass {
      var $name="Raja";
      var $obj;
      function setname($arg) {
         $this->name=$arg;
      }
      public function __clone() {
         $this->obj = clone $this->obj ;
      }
   }
   $obj1=new myclass();
   $obj1->obj=new address();
   echo "original object\n";
   print_r($obj1);
   echo "\n";

   $obj2=clone $obj1;		# cloned deep copy
   $obj1->setname("Ravi");
   $obj1->obj->setaddr("Mumbai", "400001");
   echo "after change: Original object\n";
   print_r($obj1);
   echo "\nCloned object\n";
   print_r($obj2);
?>

您現在將看到原始物件中的更改(我們將更改地址屬性)不會反映在克隆物件中,如下所示的 **輸出**:

original object
myclass Object
(
   [name] => Raja
   [obj] => address Object
   (
      [city] => Nanded
      [pin] => 431601
   )
)

after change: Original object
myclass Object
(
   [name] => Ravi
   [obj] => address Object
   (
      [city] => Mumbai
      [pin] => 400001
   )
)

Cloned object
myclass Object
(
   [name] => Raja
   [obj] => address Object
   (
      [city] => Nanded
      [pin] => 431601
   )
)
廣告