PHP4面向对象功能一个很大的缺点,是将对象视为另一种数据类型,这使得很多常见的OOP方法无法使用,如设计模式。
这些方法依赖于将对象作为引用传递给其他类方法,而不是作为值传递,而按值传递却是 PHP 的默认做法。
幸好,PHP5解决了这个问题,现在所有对象在默认情况下都被视为引用。
但是,由于所有对象都被视为引用而不是值,所以现在复制对象更为困难。如果尝试复制一个引用的对象,这只会指向原对象的地址位置。
为了解决复制问题,PHP提供了一种克隆(clone)对象的显式方法。
对象克隆
可以在对象前面加clone关键字来克隆对象,如下:
destinationObject = clone targetObject;
给出了一个详细的对象克隆示例。
这个例子使用一个名为Corporate_Drone的示例类,它包含两个成员(employeeid和tiecolor),并有相应的获取方法和设置方法。
此代码首先实例化一个corporatedrone对象,并以此为基础展示克隆操作的效果。
<?php
class corporatedrone {
private $employeeid;
private $tiecolor;
// Define a setter and getter for $employeeid
function setEmployeeID($employeeid) {
$this->employeeid = $employeeid;
}
function getEmployeeID() {
return $this->employeeid;
}
// Define a setter and getter for $tiecolor
function setTiecolor($tiecolor) {
$this->tiecolor = $tiecolor;
}
function getTiecolor() {
return $this->tiecolor;
}
}
// Create new corporatedrone object
$drone1 = new corporatedrone();
// Set the $drone1 employeeid member
$drone1->setEmployeeID("12345");
// Set the $drone1 tiecolor member
$drone1->setTiecolor("red");
// Clone the $drone1 object
$drone2 = clone $drone1;
// Set the $drone2 employeeid member
$drone2->setEmployeeID("67890");
// Output the $drone1 and $drone2 employeeid members
echo "drone1 employeeID: ".$drone1->getEmployeeID()."<br />";
echo "drone1 tie color: ".$drone1->getTiecolor()."<br />";
echo "drone2 employeeID: ".$drone2->getEmployeeID()."<br />";
echo "drone2 tie color: ".$drone2->getTiecolor()."<br />";
?>
程序运行结果:
drone1 employeeID: 12345
drone1 tie color: red
drone2 employeeID: 67890
drone2 tie color: red
可以看到,$drone2
变成一个Corporate_Drone类型的对象,并继承了$drone1
的成员值。
为进一步展示$drone2
确实是Comporate_Drone类型,这里还重新对employeeid成员进行了赋值。
__clone()
方法
可以在对象类中定义一个__clone()
方法来调整对象的克隆行为。此方法的代码将在克隆操作期间执行。
除了将所有现有对象成员复制到目标对象之外,还会执行__clone()
方法指定的操作。
下面修改Corporate_Drone类,增加以下方法:
function __clone() {
$this->tiecolor = "blue";
}
之后,创建一个新的Corporate_Drone对象,增加employeeid成员的值,克隆这个对象,然后输出一些数据,从而显示克隆对象的tiecolor确实是通过__clone()
方法设置的。
示例代码:
<?php
// Create new corporatedrone object
$drone1 = new corporatedrone();
// Set the $drone1 employeeid member
$drone1->setEmployeeID("12345");
// Clone the $drone1 object
$drone2 = clone $drone1;
// Set the $drone2 employeeid member
$drone2->setEmployeeID("67890");
// Output the $drone1 and $drone2 employeeid members
echo "drone1 employeeID: ".$drone1->getEmployeeID()."<br />";
echo "drone2 employeeID: ".$drone2->getEmployeeID()."<br />";
echo "drone2 tiecolor: ".$drone2->getTiecolor()."<br />";
?>
程序运行结果
drone1 employeeID: 12345
drone2 employeeID: 67890
drone2 tiecolor:
再来一个小例子:
<?php
class Fruit {
private $name = "水果";
private $color = "颜色";
public function setName($name){
$this->name = $name;
}
public function setColor($color){
$this->color = $color;
}
function showColor(){
return $this->color.'的'.$this->name."<br />";
}
function __destruct(){
echo "被吃掉了(对象被回收) <br />";
}
}
$apple = new Fruit();
$apple->setName("大苹果");
$apple->setColor("红色");
echo $apple->showColor();
$clone_apple = $apple;
$clone_apple->setName("小苹果");
$clone_apple->setColor("青色");
echo $clone_apple->showColor();
?>
上面只是将一个类赋值给另一个类,所以此时内存中仍是一个对象。
<?php
class Fruit {
private $name = "水果";
private $color = "颜色";
public function setName($name){
$this->name = $name;
}
public function setColor($color){
$this->color = $color;
}
function showColor(){
return $this->color.'的'.$this->name."<br />";
}
function __destruct(){
echo "被吃掉了(对象被回收) <br />";
}
function __clone(){
$this->name = "克隆水果";
}
}
$apple = new Fruit();
$apple->setName("大苹果");
$apple->setColor("红色");
echo $apple->showColor();
$clone_apple = clone $apple;
$clone_apple->setColor("青色");
echo $clone_apple->showColor();
?>
clone方法克隆出了一个新的类,所以此时内存中有两个对象。
猜你喜欢: