深入浅出理解PHP原理之变量赋值

将会对变量赋值过程中,PHP内部对数据处理的原理进行阐述,不过在讲述该原理前,需要先了解一下变量名和它的值是如何关联起来的,这个对变量赋值的理解非常重要。

PHP的变量赋值

这个标题估计很多人会不屑一顾,变量赋值?excuse me?我们学开发的第一课就会了好不好。但是,就是这样基础的东西,反而会让很多人蒙圈,比如,值和引用的关系。今天,我们就来具体讲讲。

首先,定义变量和赋值这个不用多说了吧。

  1. $a = 1;
  2. $b = '2';
  3. $c = [4, 5, 6];
  4. $d = new stdClass();

四个变量,分别定义了整型、字符串、数组的对象,这也是我们天天要打交道的四种类型。

然后,变量给变量赋值。

  1. $a1 = $a;
  2. $b1 = $b;
  3. $c1 = $c;
  4. $d1 = $d;

请注意,前三个的赋值都是正常的赋值,也就是对具体内容的拷贝,当我们修改$a1的时候$a不会有变化。$a1是新开的内存空间保存了我们的值,也就是说,他们的值是一样的,但内存地址不一样,是两个没啥关系的长得很像的人而已。

但是$d1和$d就不是了,这两货不仅值是一样的,内存地址也是一样的,这种情况就是我们所说的引用赋值,当$d1发生变化时,$d2也会产生变化。

可以这么说:引用赋值就是为原变量建立了一个Windows下的快捷方式或者Linux中的软链接。

用具体的例子来说明,首先是普通值的赋值:

  1. // 普通赋值
  2. $v = '1';
  3. $c = $v;
  4. $c = '2';
  5. echo $v, PHP_EOL; // '1'
  6. // 数组也是普通赋值
  7. $arr1 = [1,2,3];
  8. $arr2 = $arr1;
  9. $arr2[1] = 5;
  10. print_r($arr1); // [1, 2, 3]

$c不会对$v的值产生影响。$arr2修改了下标1,也就是第二个数字为5,当然也不会对$arr1产生影响。

那么对象形式的引用赋值呢?

  1. // 对象都是引用赋值
  2. class A {
  3. public $name = '我是A';
  4. }
  5. $a = new A();
  6. $b = $a;
  7. echo $a->name, PHP_EOL; // '我是A'
  8. echo $b->name, PHP_EOL; // '我是A'
  9. $b->name = '我是B';
  10. echo $a->name, PHP_EOL; // '我是B'

果然不出所料,$b修改了name属性的内容后,$a里面的name也变成了$b所修改的内容。

在这种情况下,如果对象想要不是引用传递的,一是使用__clone(),也就是原型模式来进行自己的拷贝,二是从外面重新new一个呗。

  1. // 使用克隆解决引用传递问题
  2. class Child{
  3. public $name = '我是A1的下级';
  4. }
  5. class A1 {
  6. public $name = '我是A';
  7. public $child;
  8. function __construct(){
  9. $this->child = new Child();
  10. }
  11. function __clone(){
  12. $this->name = $this->name;
  13. // new 或者用Child的克隆都可以
  14. // $this->child = new Child();
  15. $this->child = clone $this->child;
  16. }
  17. }
  18. a1 = new A1();
  19. echo $a1->name, PHP_EOL; // 输出a1原始的内容
  20. echo $a1->child->name, PHP_EOL;
  21. $b1 = $a1;
  22. echo $b1->name, PHP_EOL; // b1现在也是a1的内容
  23. echo $b1->child->name, PHP_EOL;
  24. $b1->name = '我是B1'; // b1修改内容
  25. $b1->child->name = '我是B1的下级';
  26. echo $a1->name, PHP_EOL; // a1变成b1的内容了
  27. echo $a1->child->name, PHP_EOL;
  28. // 使用__clone
  29. $b2 = clone $b1; // b2克隆b1
  30. $b2->name = '我是B2'; // b2修改内容
  31. $b2->child->name = '我是B2的下级';
  32. echo $b1->name, PHP_EOL; // b1不会变成b2修改的内容
  33. echo $b1->child->name, PHP_EOL;
  34. echo $b2->name, PHP_EOL; // b2修改的内容没问题,b1、b2不是一个货了
  35. echo $b2->child->name, PHP_EOL;

对象的引用这一块确实会容易让人蒙圈,特别是更加复杂的对象,内部的属性还有各种引用其他对象的时候,这种情况下一定要仔细确认引用赋值会不会带来问题,如果有问题,就使用新对象或者克隆技术进行引用问题的处理。

最后,轻松一下,引用变量的赋值就和我们给方法传引用参数一样的,使用一个&符号就可以啦!

  1. // 引用赋值
  2. $b = &$v;
  3. $b = '3';
  4. echo $v, PHP_EOL;

今天我们更深入的学习和了解了一下PHP中的赋值问题,特别是普通赋值和引用赋值的问题。下回看代码和框架的时候可以注意注意别人是怎么灵活使用这两种赋值的哈,自己也能试试能不能运用这两种方式改造下自己曾经写过的BUG哦!

测试代码:

https://github.com/zhangyue0503/dev-blog/blob/master/php/201910/source/PHP%E7%9A%84%E5%8F%98%E9%87%8F%E8%B5%8B%E5%80%BC.php

参考文档:

https://www.php.net/manual/zh/language.variables.basics.php