在很多系统的权限/选项设置中 很多都用到了位运算的方法来存储多种标志位。这样可以节省字段。一个字段只需要一个数字 就可以标识很多种设置和信息。 举例 dicuz的帖子表的status字段,官方预留了16个标志位(0x0000 - 0xFFFF) 即2 16 目前规划使用了只有8个
在很多系统的权限/选项设置中 很多都用到了位运算的方法来存储多种标志位。这样可以节省字段。一个字段只需要一个数字 就可以标识很多种设置和信息。
举例 dicuz的帖子表的status字段,官方预留了16个标志位(0x0000 - 0xFFFF) 即216
目前规划使用了只有8个标志位,如下
?0000 0000 0000 0001 是否缓存帖子位置信息 0000 0000 0000 0010 是否回帖只对管理人员和发帖者可见 0000 0000 0000 0100 是否抢楼贴 0000 0000 0000 1000 是否倒序查看回帖 0000 0000 0001 0000 是否存在主题图章标志位 0000 0000 0010 0000 回复是否通知作者 0000 0000 0100 0000 是否推送到QQ空间 0000 0000 1000 0000 是否推送到腾讯微博
这8种状态可以使用一个数字来同时表示,节省了字段
那么这种东西的原理是什么呢
立即学习“PHP免费学习笔记(深入)”;
这个我们可以复习一下的位运算单算法
| 例子 | 名称 | 结果 |
|---|---|---|
| $a & $b | And(按位与) | 将把 $a 和 $b 中都为 1 的位设为 1。 |
| $a | $b | Or(按位或) | 将把 $a 或者 $b 中为 1 的位设为 1。 |
| $a ^ $b | Xor(按位异或) | 将把 $a 和 $b 中不同的位设为 1。 |
| ~ $a | Not(按位非) | 将 $a 中为 0 的位设为 1,反之亦然。 |
| $a | Shift left(左移) | 将 $a 中的位向左移动 $b 次(每一次移动都表示“乘以 2”)。 |
| $a >> $b | Shift right(右移) | 将 $a 中的位向右移动 $b 次(每一次移动都表示“除以 2”)。 |
比如
14 = 0b1110
11 = 0b1011
那么 14 & 11? = 0b1110 & 0b1011 = 0b1010 = 10
还是上面那个例子
14 | 11 = 0b1110 | 0b1011 = 0b1111 = 15
14 ^ 11 = 0b1110 ^ 0b1011 = 0b0101 = 5
非运算比较特殊 涉及到符号 这里要说一下补码 反码 原码的概念
1.二进制最高是符号位? 0是正数? 1表示负数
2.正数的 原码 反码? 补码 都一样(我上面没有单独算补码的原因 ,正数补码和反码一样)
帮管客CRM客户管理系统查看详情基于WEB的企业计算,php+MySQL进行开发,性能稳定可靠,数据存取集中控制,避免了数据泄漏的可能,采用加密数据传递参数,保护系统数据安全,多级的权限控制,完善的密码验证与登录机制更加强了系统安全性。
1377
![]()
3.用二进制表示一个数? 这个码 就是原码 比如 14的原码就是 1110
4.负数的反码 等于? 他符号位不变 其他取反,而负数的补码等于他的反码+1
5.计算机运算的时候 全都是以补码的形式来运算的 不管正负数
那么
1 是正数,那么他的原码 0001 = 反码 = 补码 = 0001 =>取反 后补码1110
那么这个符号位是1就是负数 也就是010代表的负数就是-2 也就是 ~1 = -2
1
1的补码??? 00000001
移动2位??? 00000100
正数的反码 补码? 原码 都一样 所以 是个4
负数的计算过程相同 不再赘述 左移也类似 4>>2? 就是1
其实可以理解为右移在十进制的表现上就是乘以2 左移 在十进制的表现上就是除以2
那么回到本文正题 如何用一个数字来标识这些权限位呢?
以刚才discuz的帖子表达status字段为例,检查帖子回复是否通知作者 就看二进制上第六位是否是置位为1 那么怎么检查呢?就是用上面我们提到的与运算。
与运算是将把 $a 和 $b 中都为 1 的位设为 1。那么假设
$a=36=0b 0010 0100
$b=0b 0010 0000
$a&$b = 0b 0010 0100 & 0b 0010 0000 = 0b 0010 0000 = 32 = 26-1 = 25
因此 检查,某个数代表的第N个权限标志位有没有置位(是1) 只要选择该数与仅该标志位置位的操作数2N-1进行与运算即可,相反要计算某个标志位被置位的数字 只要选择合适的操作数进行或运算即可。我们可以看discuz对此的实现:
function getstatus($status, $position) {
$t = $status & pow(2, $position - 1) ? 1 : 0;
return $t;
}
function setstatus($position, $value, $baseon = null) {
$t = pow(2, $position - 1);
if($value) {
$t = $baseon | $t;
} elseif ($baseon !== null) {
$t = $baseon & ~$t;
} else {
$t = ~$t;
}
return $t & 0xFFFF;
}注意 写这段代码的人显然受到了C的影响 其实 $a & ~$b 和 $a ^ $b 是等效的 只不过 ^是PHP的写法 另外 pow(2, $position - 1)换成 1
原文地址:PHP中的二进制位运算和权限存储, 感谢原作者分享。
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号