在php5 的对象编程经常提到的一个关键点是“默认情况下对象是通过引用传递的”。但其实这不是完全正确的。下面通过一些例子来说明。
php的引用是别名,就是两个不同的变量名字指向相同的内容。在php5,一个对象变量已经不再保存整个对象的值。只是保存一个标识符来访问真正的对象内容。 当对象作为参数传递,作为结果返回,或者赋值给另外一个变量,另外一个变量跟原来的不是引用的关系,只是他们都保存着同一个标识符的拷贝,这个标识符指向同一个对象的真正内容。
通俗说:普通的赋值操作只是传递了索引,只有在清除传值对象的时候,才会发生写时复制
<?php
//深入理解PHP对象赋值
echo '<pre>';
$obj = new StdClass;
$obj->name = 'Pig';
var_dump($obj); //object(stdClass)#1 (1) { ["name"]=> string(3) "Pig" }
$copy = $obj; // $obj ,$copy都是new StdClass返回的同一个标识符的拷贝
var_dump($copy); //object(stdClass)#1 (1) { ["name"]=>string(3) "Pig" }
$objRef = &$obj; // 此时会将$obj转换成引用,然后赋值给$objRef,因此$obj,$objRef都为引用
var_dump($objRef); //object(stdClass)#1 (1) { ["name"]=>string(3) "Pig" }
$objClone = clone $obj; //新空间
$obj->name = 'After Clone';
var_dump($objClone); //object(stdClass)#1 (1) { ["name"]=>string(3) "Pig" }
var_dump($obj); //object(stdClass)#1 (1) { ["name"]=>string(11) "After Clone" }
//unset是删除引用效果
$nameRef = &$obj->name; //$obj->name被转换成引用(& string),然后赋给$nameRef
var_dump($obj); //object(stdClass)#2 (1) { ["name"]=>&string(11) "After Clone" }
unset($nameRef); //删除引用
var_dump($obj); //object(stdClass)#1 (1) { ["name"]=>string(11) "After Clone" }
//null是赋值效果
$nameRef = &$obj->name; //恢复name的引用
var_dump($obj); //object(stdClass)#2 (1) { ["name"]=>&string(11) "After Clone" }
$nameRef = null;
var_dump($obj); //object(stdClass)#2 (1) { ["name"]=>&NULL }
unset($objRef); //仅仅删除了引用
var_dump($obj); //object(stdClass)#1 (1) { ["name"]=>&NULL }
$objRef = &$obj; //恢复对象引用
$obj->name = 'Lucy';
$obj = null; //赋值$obj为null,$obj只是new StdClass的标识拷贝,不会影响其内容。
//$objRef做为$obj的引用,会同时被赋值null
//等价 $objRef = null;
var_dump($obj,$copy,$objRef,$objClone);
// NULL,
// object(stdClass)#1 (1) { ["name"]=>&string(4) "Lucy" },
// NULL,
// object(stdClass)#1 (1) { ["name"]=>string(3) "Pig" }
?>
对象克隆
有的时候我们需要在一个项目里面使用两个或多个一样的对象,如果使用 new 关键字重新创建对象,再赋值上相同的属性,这样做比较烦琐而且也容易出错。PHP 提供了对象克隆功能,可以根据一个对象完全克隆出一个一模一样的对象,而且克隆以后,两个对象互不干扰。
使用关键字 clone 来克隆对象。语法:
$object2 = clone $object;
1
例子:
<?php
class Person {
private $name;
private $age;
function __construct($name, $age) {
$this->name=$name;
$this->age=$age;
}
function say() {
echo "我的名字叫:".$this->name."<br />";
echo "我的年龄是:".$this->age;
}
}
$p1 = new Person("张三", 20);
$p2 = clone $p1;
$p2->say();
?>
运行例子,输出:
我的名字叫:张三
我的年龄是:20
1
2
__clone()
如果想在克隆后改变原对象的内容,需要在类中添加一个特殊的 __clone() 方法来重写原本的属性和方法。__clone() 方法只会在对象被克隆的时候自动调用。
例子:
<?php
class Person {
private $name;
private $age;
function __construct($name, $age) {
$this->name = $name;
$this->age = $age;
}
function say() {
echo "我的名字叫:".$this->name;
echo " 我的年龄是:".$this->age."<br />";
}
function __clone() {
$this->name = "我是假的".$this->name;
$this->age = 30;
}
}
$p1 = new Person("张三", 20);
$p1->say();
$p2 = clone $p1;
$p2->say();
?>
运行例子,输出:
我的名字叫:张三 我的年龄是:20
我的名字叫:我是假的张三 我的年龄是:30
————————————————
版权声明:本文为优快云博主「Shower稻草人」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.youkuaiyun.com/u013474436/article/details/50773646