PHP中,=
的作用都是将一个值复制给另一个(大多数编程语言都是一样),将=
作用在基本数据类型上时,就直接进行了赋值,并且变量的修改互不影响,如下:
$a = 1;
$b = $a;
$b = 2;
print_r($a); // 1
print_r($b); // 2
而在复制对象时,=
只是简单地将两个变量指向同一个类实例,测试一下:
class Student
{
public $name;
public $age;
function __construct($name, $age) {
$this->name = $name;
$this->age = $age;
}
}
$student1 = new Student('mike', 19);
$student2 = $student1;
print_r($student1);
print_r($student2);
var_dump($student1 == $student2);
var_dump($student1 === $student2);
// 输出
Student Object
(
[name] => mike
[age] => 19
)
Student Object
(
[name] => mike
[age] => 19
)
bool(true)
bool(true)
从上面的代码中,就能明显看出$student1
和$student2
两个变量指向的是同一个对象,新手看到这可能并没有发现其中蹊跷,而且等值检测也没有问题啊(其实这里的等值检测是看不出什么的),这时,如果我继续修改$student2
中的属性时,就能发现问题:
$student2->name = 'Jack';
$student2->age = 22;
print_r($student1);
print_r($student2);
// 输出
Student Object
(
[name] => Jack
[age] => 22
)
Student Object
(
[name] => Jack
[age] => 22
)
为什么会这样呢,我只是想修改$student2
的属性,为什么$student1
也改了,这就好比Dota里面的地卜师,只要逮到你一个分身,你就挂了。
![](https://img.haomeiwen.com/i2650331/ccd5e64f18eb249f.png)
其实,看到这种情况,大概就能猜到PHP中对象的赋值都是通过引用操作,$student1
和$student2
没有各自保留一份单独的副本,在内存中的分配大概就是这样的:
![](https://img.haomeiwen.com/i2650331/983e9b2d3be242b1.png)
两个变量指向的是同一个内存地址,所以我们改变其中一个变量的属性是另一个也会变,但如果我们想要对两个变量做不同的操作并且互不影响该怎么办?
好在PHP提供了一个clone
关键字来达到这个目的。
$student1 = new Student('mike', 19);
$student2 = clone $student1;
$student2->name = 'Jack';
$student2->age = 22;
print_r($student1);
print_r($student2);
// 输出
Student Object
(
[name] => mike
[age] => 19
)
Student Object
(
[name] => Jack
[age] => 22
)
通过clone
复制后,现在$student1
和$student2
就是两个不同的变量,修改互不影响。
关于__clone()方法
现在有这么一种情况,在复制对象时,想做一些修改操作,实际开发中,$id
属性可能会与数据库表中某条记录一一对应,在复制之后,两个对象就指向数据库中的同一条记录了,那么使用__clone()
就能控制复制时,哪些属性可以复制,哪些应该不复制或者置空。__clone()
是个魔术方法,在对象上调用clone
关键字时会自动调用。如下代码所示:
class Student
{
public $id;
public $name;
public $age;
function __construct($id, $name, $age) {
$this->id = $id;
$this->name = $name;
$this->age = $age;
}
function __clone() {
$this->id = 0;
}
}
$student1 = new Student(1,'mike', 19);
$student2 = clone $student1;
print_r($student1);
print_r($student2);
// 输出
Student Object
(
[id] => 1
[name] => mike
[age] => 19
)
Student Object
(
[id] => 0
[name] => mike
[age] => 19
)
当在$student1
上调用clone
时,产生一个新的副本给$student2
,这时$student2
上就会自动调用__clone
方法,使$id
为0。
上面通过clone
和__clone()
可以保证所有的基础数据类型可以被完全复制,但如果复制的对象中的属性也是一个对象时,复制后的两个对象都会引用同一个属性,修改时会互相影响,如下:
Class Score {
public $english;
public $math;
function __construct($english, $math) {
$this->english = $english;
$this->math = $math;
}
}
class Student
{
public $id;
public $name;
public $age;
public $score;
function __construct($id, $name, $age, Score $score) {
$this->id = $id;
$this->name = $name;
$this->age = $age;
$this->score = $score;
}
function __clone() {
$this->id = 0;
}
}
$student1 = new Student(1,'mike', 19, new Score(80, 90));
$student2 = clone $student1;
// 我们需要修改$student1的英语成绩
$student1->score->english = 99;
//结果$student2的英语成绩也被修改了
echo $student2->score->english;
// 输出
99
上面这种情况并不是我们希望看到的,我们不想对象属性复制后被共享。解决方法就是在__clone()
做做一些修改:
function __clone() {
$this->id = 0;
$this->score = clone $this->score;
}
// 输出
80
我所了解的PHP中对象复制大概就是这样,类似__clone()
这样的魔术方法还有好几个,常见的还有:__set()
、__unset()
、__call()
等,利用这些魔术方法可以实现更多复杂的操作,以后慢慢学习。
网友评论