最近接手了五六年前的老项目,用的是CI2.0框架,在看框架源码的时候有个地方让我楞了一下,于是有了这篇文章。
字符&
的最早历史可以追溯到公元1世纪,最早是拉丁语et (意为and)的连写。
&
是指逻辑上表示两者属于缺一不可的关系,还表示意思是一个人和另外一个人之意,与and同义。
&
在PHP项目中是经常使用的一个操作符, 例如按位与丶逻辑操作丶引用变量丶引用传递丶引用返回。
$a & $b
将把 $a
和 $b
中都为1的位设为1.
整数与1
进行按位与运算,运算结果为1
表示为奇数, 运算结果为0
表示为偶数。例如:
5和1进行按位与&
, 得到结果为1。
十进制 | 二进制 |
---|---|
5 | 101 |
1 | 001 |
6和1进行按位与&
, 得到结果为0。
十进制 | 二进制 |
---|---|
6 | 110 |
1 | 001 |
但是项目中我们都不会这么写, 都是使用n % 2 == 0
, 因为奇偶性判断使用%
的效率比较高,也比较容易理解。
假设有个系统中有用户权限分配模块, 其权限设置如下:
权限名称 | 权限值 |
---|---|
查看 | 1 |
新增 | 2 |
修改 | 4 |
删除 | 8 |
那么存储该用户的权限不需要存储逗号分割的字符串1,2,4,8
, 只需要存储一个整形:15
即可。
如果用户1拥有新增丶查看丶修改权限, 那么需要存储的权限值为: 1+2+4=7
。
如果用户2拥有所有权限, 那么需要存储的权限值为:1+2+4+8=15
。
检查用户1是否有修改权限:7 & 4
结果为4
, 表示拥有修改权限。
检查用户2是否有删除权限:15 & 8
结果为8
, 表示拥有删除权限。
$a && $b
只有$a
和$b
均为真,结果才为真。
PHP的引用允许你用两个变量来指向同一个内容。无论对哪个变量名的值进行了修改,其他变量名访问的内容也会随之改变。
与C语言中的指针是有差别的。C语言中的指针里面存储的是变量的内容,在内存中存放的地址。
<?php $a = 10; $b = &$a; $a = 11; var_dump($a, $b); // 输出11, 11 $b = 12; var_dump($a, $b); // 输出12, 12
foreach循环时加&
符号会有什么问题? 例如以下代码能正常输出期望值吗?
<?php $values = ['Python', 'Php', 'Go']; foreach ($values as &$value) { $value = strtoupper($value); } foreach ($values as $value){ echo $value . PHP_EOL; }
预期结果是: ['PYTHON', 'PHP', 'GO']
实际结果是['PYTHON', 'PHP', 'PHP']
。
这是因为第一次foreach
中使用了引用, 使$value
成为了引用变量, 并且$arr[2]
和$value
指向了同一个地址空间(共享变量值), 第二次foreach
时, 不断的将$values
中的值赋给$value
, 导致$arr[2]
的值也被修改了。
可以将一个变量通过引用传递给函数,这样该函数就可以修改其参数的值。
/** * 对传入的值进行平方 * Author: ClassmateLin * Email: classmatelin.site@gmail.com * Site: https://www.classmatelin.top * @param $n */ function f(&$n) { $n *= $n; } $n = 2; f($n); var_dump($n); // 输出4
按值传递是需要对变量进行拷贝, 引用传递是同一内存空间。
如果是大型字符串或对象,那么使用引用传递比按值传递可以节省一些内存, 但是使用引用传递代码可读性稍微低点。
函数的引用返回,在方法前加&
符号定义。同时需要接收返回值也需要&
, 否则将不起作用, 例如:
<?php class Foo { public $value = 10; /** * 返回value, 引用返回, 对返回的值修改会影响到该值。 * Author: ClassmateLin * Wechat: ClassmateLin_ * Email: classmatelin.site@gmail.com * Site: https://www.classmatelin.top * @return int */ public function &getValue() { return $this->value; } } $foo = new Foo(); $val1 = $foo->getValue(); $val1 = 11; // $val1没有用&接收,不是引用, 修改不会影响实例$foo的值. var_dump($val1, $foo->getValue()); // 输出11, 10 $val2 = &$foo->getValue(); // $val2是引用 $val2 = 13; var_dump($val2, $foo->getValue()); // 输出13, 13