
PHP 里怎么写二进制字面量?
PHP 从 5.4 开始支持直接用 0b 前缀写二进制整数,比如 0b1010 就是十进制的 10。不用函数、不靠字符串转换,就是原生语法。
常见错误现象:0b102 报错 —— 二进制只允许 0 和 1,出现 2 就直接 Parse error;b1010 或 0B1010(大写 B)在老版本(
-
0b必须小写,0B在部分旧环境不识别 - PHP 5.3 及更早版本不支持,必须升级或改用
bindec() - 不能带空格或下划线:
0b10_10(PHP 7.4+ 才支持下划线分隔符),老版本会解析失败
bindec() 函数怎么用?什么情况下必须用它?
bindec() 是把二进制字符串转成整型的唯一内置函数,适用于运行时拼接、用户输入、配置读取等无法用字面量的场景。
使用场景:读取 JSON 配置里的 "flags": "10101",或表单提交的 $_POST['mask'] = '111' —— 这些都是字符串,没法加 0b 前缀。
立即学习“PHP免费学习笔记(深入)”;
- 参数必须是纯数字字符串,
bindec('1010')→ 10;bindec('0b1010')→ 0(开头非数字,直接截断失败) - 遇到非法字符(如空格、字母)会静默截断,
bindec('101x0')实际算的是bindec('101')→ 5,容易埋坑 - 性能上比字面量慢一个数量级,但日常业务几乎感知不到;高频循环里建议预计算或缓存
为什么 (int) '0b1010' 不行?
因为 PHP 的类型强制转换不识别二进制前缀。(int) '0b1010' 会按十进制解析字符串,结果是 0(遇到 b 就停了)。
同理,intval('0b1010') 默认按十进制转,除非显式指定进制:intval('1010', 2) 才等价于 bindec('1010')。
-
intval($str, 2)和bindec($str)行为一致,但intval()第二个参数不传时默认是 10,极易漏写 -
filter_var($str, FILTER_VALIDATE_INT, ['flags' => FILTER_FLAG_ALLOW_OCTAL | FILTER_FLAG_ALLOW_HEX])不支持二进制,别试 - 正则校验字符串是否为合法二进制再转,比直接
bindec()多一重判断,但能提前暴露脏数据
注意整型溢出和符号位问题
PHP 整型是平台相关的(通常是 64 位有符号),最大正数是 9223372036854775807(0b111...111 共 63 个 1)。超了会自动转成 float,精度丢失。
比如 0b1000000000000000000000000000000000000000000000000000000000000000(64 位全 1)在 64 位系统上不是 -1,而是 float,且值不对。
- 需要处理高位掩码或协议字段时,优先用
gmp_init($str, 2)或bcadd()等扩展,避免溢出 - PHP 没有无符号整型关键字,
0b10000000在 8 位上下文里可能是 128,也可能是 -128(取决于你怎么解释),语言层不保证 - 跟 C/嵌入式交互时,别假设 PHP 能“原样”表示某个二进制位模式,得靠 pack/unpack 显式控制字节序和符号
二进制字面量看着简单,但混着运行时数据、老版本兼容、溢出边界一起用,就很容易在某个凌晨三点的生产环境里突然翻车。最稳妥的做法是:能用 0b 就用,不能用就上 bindec() + 字符串校验,别信强制转换,也别手写位运算模拟。











