
php 中对象赋值默认为引用:正确克隆对象避免意外修改
在 PHP 中,对象(object)本质上是引用类型——这与数组、标量(如 int、string)的行为截然不同。当你执行 $new_item = $item; 时,并未创建一个独立的新对象,而只是让 $new_item 指向与 $item 完全相同的内存地址中的同一个对象实例。因此,对 $new_item->title 的任何修改,都会实时反映在 $item 上——这正是你观察到“原对象也被改变”的根本原因。
以下代码清晰展示了该机制:
$item = (object)['id' => 2, 'title' => 'original 2']; $new_item = $item; // ❌ 不是复制,而是引用绑定 $new_item->title = 'new 2'; var_dump($item->title); // string(6) "new 2" var_dump($new_item->title); // string(6) "new 2"
✅ 正确做法:使用 clone 创建独立副本
clone 是 PHP 提供的专用操作符,用于生成对象的浅拷贝(shallow copy)——即新建一个具有相同属性值的对象实例,且与原对象互不干扰:
$item = (object)['id' => 2, 'title' => 'original 2']; $clone = clone $item; // ✅ 创建全新对象实例 $clone->title = 'new 2'; var_dump($item->title); // string(12) "original 2" var_dump($clone->title); // string(6) "new 2"
回到你的实际场景,修复方案如下(已优化逻辑与可读性):
立即学习“PHP免费学习笔记(深入)”;
foreach ($calendar as $key => $item) {
if ($item->id == 2) {
$clone = clone $item; // 创建独立副本
$clone->title = 'new 2'; // 仅修改副本
// 在原位置插入副本(保持顺序:原项后移)
array_splice($calendar, $key, 0, [$clone]);
break; // 避免重复处理(可选)
}
}⚠️ 注意事项:
- clone 是浅拷贝:若对象属性本身是对象或数组,其内部引用关系仍被共享。如需深度克隆,需在类中实现 __clone() 魔术方法手动处理嵌套对象。
- foreach 中的 $item 默认是值拷贝(对对象而言即引用拷贝),绝非引用传递——除非你显式声明 foreach ($arr as &$item)。本例中未加 &,问题根源不在循环语法,而在对象赋值语义本身。
- 函数参数传入对象时同样遵循引用语义(PHP 5+),无需引用符号即可在函数内修改原始对象——这是语言设计特性,而非 bug。
总结:理解 PHP 对象的引用本质,是写出健壮面向对象代码的前提。当需要副本时,请坚定使用 clone;若需彻底隔离复杂嵌套结构,务必配合 __clone() 实现自定义深拷贝逻辑。











