我一直认为,基础数据类型都是传值调用,而具体的类都是传引用调用,
其实没有传引用调用,所有的函数参数都是传值调用(除了PHP的&或指针调用)
1
$a = 1;
test($a);
echo $a;
function test($a) {
$a = 2;
}
输出的是 1
2
$a = new Man;
$a->age = 1;
test($a)
echo $a->age;
class Man {
public $age;
}
function test(Man $a) {
$a->age = 2;
}
这里输出是2
3
$a = new Man;
$a->age = 1;
test($a);
echo $a->age;
class Man {
public $age;
}
function test(Man $a) {
$a = new Man;
$a->age = 2;
}
输出是 1
其实所有都是传值,只不过,这个值是一个类,这个类是个拷贝的,但是拷贝类的内部成员的地址都是原类的成员的真实地址,而他自己确实是拷贝的,
换一种方法理解
$a = new Man;
$a->age = 1;
$b = $a;
$b->age = 3;
echo $a->age;
class Man {
public $age;
}
这里输出 3
$a = new Man;
$a->age = 1;
$b = $a;
$b = new Man;
$b->age = 3;
echo $a->age;
class Man {
public $age;
}
这里输出 1
这样就好理解了,实际上函数的调用传参和这种赋值是一个道理
当 $b = new Man; 的时候, a的zval就要发生分裂了(参见PHP引用计数)
而 如果没有 $b = new Man; 直接 $b->age = 3,实际上$b->age相当与 &$a->age,也就是说不会发生zval的分裂
结论,类参数传递是也是一个普通的值(拷贝)传递,只不过类拷贝的成员变量都是源类的成员变量的引用,所以你直接修改类如
function test(Man $a) {
$a = new Man;
}
是不会有任何效果的,当然强制引用传递和指针除外,例如下面的
function test(Man &$a) {
$a = new Man;
}
The End