web-dev-qa-db-ja.com

PHPでオブジェクトのコピーを作成するにはどうすればよいですか?

PHPオブジェクトでは参照によって渡されるようです。代入演算子でさえ、オブジェクトのコピーを作成しているようには見えません。

これは、単純で、人為的な証拠です。

<?php

class A {
    public $b;
}


function set_b($obj) { $obj->b = "after"; }

$a = new A();
$a->b = "before";
$c = $a; //i would especially expect this to create a copy.

set_b($a);

print $a->b; //i would expect this to show 'before'
print $c->b; //i would ESPECIALLY expect this to show 'before'

?>

両方の印刷ケースで、私は「後」を得ています

したがって、$ aset_b()を渡し、参照ではなく?

143
Nick Stinemates

PHPでは、5 +オブジェクトが参照によって渡されます。 PHP 4では、それらは値によって渡されます(そのため、ランタイムによる参照渡しが行われたため、廃止されました)。

PHP5で「clone」演算子を使用して、オブジェクトをコピーできます。

$objectB = clone $objectA;

また、質問で言ったように、参照によって渡されるのはオブジェクトだけです...

244
Eran Galperin

回答は一般的にJava本にあります。

  1. クローン作成:cloneメソッドをオーバーライドしない場合、デフォルトの動作はシャローコピーです。オブジェクトにプリミティブなメンバー変数しかない場合は、まったく問題ありません。しかし、メンバー変数として別のオブジェクトを使用するタイプレス言語では、頭痛の種です。

  2. シリアライゼーション/デシリアライゼーション

$new_object = unserialize(serialize($your_object))

これにより、オブジェクトの複雑さに応じて、高コストでディープコピーが実現します。

92
yogman

前のコメントによると、メンバー変数として別のオブジェクトがある場合は、次を実行します。

class MyClass {
  private $someObject;

  public function __construct() {
    $this->someObject = new SomeClass();
  }

  public function __clone() {
    $this->someObject = clone $this->someObject;
  }

}

これでクローンを作成できます:

$bar = new MyClass();
$foo = clone $bar;
19
Stanislav

ドキュメントによると( http://ca3.php.net/language.oop5.cloning ):

$a = clone $b;
9
Leo

PHPがコピーオンライトを使用することを明確にするため、変更するまでは基本的にすべてが参照ですが、オブジェクトについては、受け入れられた答えのようにcloneと__clone()マジックメソッドを使用する必要があります。

4
Patricio Rossi

このコードはメソッドのクローン作成に役立ちます

class Foo{

    private $run=10;
    public $foo=array(2,array(2,8));
    public function hoo(){return 5;}


    public function __clone(){

        $this->boo=function(){$this->hoo();};

    }
}
$obj=new Foo;

$news=  clone $obj;
var_dump($news->hoo());
1
zloctb

私はいくつかのテストを行っていて、これを得ました:

class A {
  public $property;
}

function set_property($obj) {
  $obj->property = "after";
  var_dump($obj);
}

$a = new A();
$a->property = "before";

// Creates a new Object from $a. Like "new A();"
$b = new $a;
// Makes a Copy of var $a, not referenced.
$c = clone $a;

set_property($a);
// object(A)#1 (1) { ["property"]=> string(5) "after" }

var_dump($a); // Because function set_property get by reference
// object(A)#1 (1) { ["property"]=> string(5) "after" }
var_dump($b);
// object(A)#2 (1) { ["property"]=> NULL }
var_dump($c);
// object(A)#3 (1) { ["property"]=> string(6) "before" }

// Now creates a new obj A and passes to the function by clone (will copied)
$d = new A();
$d->property = "before";

set_property(clone $d); // A new variable was created from $d, and not made a reference
// object(A)#5 (1) { ["property"]=> string(5) "after" }

var_dump($d);
// object(A)#4 (1) { ["property"]=> string(6) "before" }

?>
1
Pyetro

この例では、iPhoneクラスを作成し、cloningによって正確なコピーを作成します

class iPhone {

public $name;
public $email;

    public function __construct($n, $e) {

       $this->name = $n;
       $this->email = $e;

    }
}


$main = new iPhone('Dark', '[email protected]');
$copy = clone $main;


// if you want to print both objects, just write this    

echo "<pre>"; print_r($main);  echo "</pre>";
echo "<pre>"; print_r($copy);  echo "</pre>";
0