PHP中的对象赋值操作和其他普通变量的赋值逻辑有明显区别,普通变量赋值会直接拷贝值生成新的独立变量,而对象赋值默认采用的是引用传递的语义,这往往会让刚接触PHP的开发者感到困惑。

PHP对象赋值的默认行为
在PHP中,当我们把一个对象赋值给另一个变量时,并不是直接复制整个对象的内容,而是复制对象的标识符,这个标识符指向同一个对象实例。我们可以通过一个简单的示例来验证这个特性:
<?php
class User {
public $name;
public $age;
}
// 创建第一个对象实例
$user1 = new User();
$user1->name = '张三';
$user1->age = 20;
// 将$user1赋值给$user2
$user2 = $user1;
// 修改$user2的属性
$user2->name = '李四';
$user2->age = 25;
// 输出两个对象的属性
echo 'user1的name:' . $user1->name . PHP_EOL; // 输出 李四
echo 'user1的age:' . $user1->age . PHP_EOL; // 输出 25
echo 'user2的name:' . $user2->name . PHP_EOL; // 输出 李四
echo 'user2的age:' . $user2->age . PHP_EOL; // 输出 25
?>
从上面的代码可以看到,修改$user2的属性之后,$user1的属性也发生了同样的变化,这说明两个变量指向的是同一个对象实例,这就是PHP对象赋值默认引用传递的表现。
为什么PHP对象赋值默认是引用传递
PHP的底层设计中,对象本身存储在堆内存中,而变量保存的是指向这个堆内存地址的引用。当进行对象赋值的时候,只是把引用值复制给了新的变量,两个变量最终指向的是同一个堆内存中的对象,所以修改任意一个变量的属性,都会影响到另一个变量。这种设计可以减少不必要的内存开销,避免每次赋值都复制整个对象占用的内存空间。
clone关键字的作用
如果我们想要复制一个全新的对象,让新对象和原对象完全独立,互不影响,就需要使用clone关键字。clone会创建一个新的对象实例,把原对象的属性值复制到新对象中。我们修改上面的示例,使用clone来赋值:
<?php
class User {
public $name;
public $age;
}
$user1 = new User();
$user1->name = '张三';
$user1->age = 20;
// 使用clone复制对象
$user2 = clone $user1;
// 修改$user2的属性
$user2->name = '李四';
$user2->age = 25;
// 输出两个对象的属性
echo 'user1的name:' . $user1->name . PHP_EOL; // 输出 张三
echo 'user1的age:' . $user1->age . PHP_EOL; // 输出 20
echo 'user2的name:' . $user2->name . PHP_EOL; // 输出 李四
echo 'user2的age:' . $user2->age . PHP_EOL; // 输出 25
?>
这次修改$user2的属性之后,$user1的属性没有发生变化,说明两个对象是相互独立的,这就是clone的作用。
__clone魔术方法的使用
如果对象中有引用类型的属性,比如属性本身是另一个对象,那么默认使用clone只会进行浅拷贝,也就是只复制引用,不会复制引用指向的对象。这时候我们可以通过__clone魔术方法来定义深拷贝的逻辑。示例如下:
<?php
class Address {
public $city;
}
class User {
public $name;
public $address;
public function __clone() {
// 对address属性进行深拷贝
$this->address = clone $this->address;
}
}
$address = new Address();
$address->city = '北京';
$user1 = new User();
$user1->name = '张三';
$user1->address = $address;
$user2 = clone $user1;
// 修改$user2的address属性
$user2->address->city = '上海';
echo 'user1的address city:' . $user1->address->city . PHP_EOL; // 输出 北京
echo 'user2的address city:' . $user2->address->city . PHP_EOL; // 输出 上海
?>
上面的代码中,User类的__clone方法里对address属性进行了克隆操作,这样clone用户对象的时候,地址对象也会被复制一份,两个用户的地址对象就相互独立了。
使用场景总结
在实际开发中,我们需要根据需求选择是否使用clone:
- 如果希望多个变量操作同一个对象实例,直接使用赋值即可,不需要额外处理。
- 如果需要一个和原对象完全独立的新对象,修改新对象不会影响原对象,就需要使用
clone关键字。 - 如果对象包含其他对象的引用属性,需要深拷贝的话,就需要在类中定义
__clone方法来实现自定义的拷贝逻辑。
理解PHP对象赋值默认引用传递的特性,以及clone的正确用法,可以帮助我们避免很多因为对象操作导致的隐蔽bug,写出更健壮的PHP代码。