189

似乎在 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'

?>

在这两种印刷情况下,我都在“追求”

那么,如何通过值而不是引用将$a传递给set_b() ?

4

9 回答 9

317

在 PHP 5+ 中,对象是通过引用传递的。在 PHP 4 中,它们是按值传递的(这就是它在运行时通过引用传递的原因,这已被弃用)。

您可以使用 PHP5 中的“克隆”运算符来复制对象:

$objectB = clone $objectA;

此外,它只是通过引用传递的对象,而不是您在问题中所说的所有内容......

于 2008-10-09T04:24:09.790 回答
117

答案通常在 Java 书籍中找到。

  1. 克隆:如果不覆盖克隆方法,则默认行为是浅拷贝。如果您的对象只有原始成员变量,那完全可以。但是在以另一个对象作为成员变量的无类型语言中,这是一个令人头疼的问题。

  2. 序列化/反序列化

$new_object = unserialize(serialize($your_object))

这取决于对象的复杂性,以高成本实现深度复制。

于 2008-10-09T05:28:54.947 回答
25

根据之前的评论,如果您有另一个对象作为成员变量,请执行以下操作:

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;
于 2008-10-09T06:53:07.820 回答
11

根据文档(http://ca3.php.net/language.oop5.cloning):

$a = clone $b;
于 2008-10-09T04:23:05.157 回答
4

只是为了澄清 PHP 在写入时使用复制,所以基本上所有内容都是参考,直到您修改它,但是对于对象,您需要使用 clone 和 __clone() 魔术方法,就像在接受的答案中一样。

于 2015-05-06T00:36:36.520 回答
1

此代码帮助克隆方法

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());
于 2013-10-03T20:25:32.220 回答
1

我正在做一些测试并得到了这个:

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" }

?>
于 2014-01-23T19:29:00.943 回答
1

在这个例子中,我们将创建iPhone类并通过克隆从它复制

class iPhone {

public $name;
public $email;

    public function __construct($n, $e) {

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

    }
}


$main = new iPhone('Dark', 'm@m.com');
$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>";
于 2017-11-03T16:59:36.040 回答
-1

如果您想在不同实例中完全复制对象的属性,您可能需要使用此技术:

将其序列化为 JSON,然后将其反序列化回 Object。

于 2013-08-20T14:00:47.727 回答