&
符号你真的了解吗?最近接手了五六年前的老项目,用的是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语言中的指针里面存储的是变量的内容,在内存中存放的地址。
$a = 10;
$b = &$a;
$a = 11;
var_dump($a, $b); // 输出11, 11
$b = 12;
var_dump($a, $b); // 输出12, 12
foreach循环时加&
符号会有什么问题? 例如以下代码能正常输出期望值吗?
$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: [email protected]
* Site: https://www.classmatelin.top
* @param $n
*/
function f(&$n)
{
$n *= $n;
}
$n = 2;
f($n);
var_dump($n); // 输出4
按值传递是需要对变量进行拷贝, 引用传递是同一内存空间。
如果是大型字符串或对象,那么使用引用传递比按值传递可以节省一些内存, 但是使用引用传递代码可读性稍微低点。
函数的引用返回,在方法前加&
符号定义。同时需要接收返回值也需要&
, 否则将不起作用, 例如:
class Foo {
public $value = 10;
/**
* 返回value, 引用返回, 对返回的值修改会影响到该值。
* Author: ClassmateLin
* Wechat: ClassmateLin_
* Email: [email protected]
* 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