我一直认为,基础数据类型都是传值调用,而具体的类都是传引用调用,
其实没有传引用调用,所有的函数参数都是传值调用(除了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