阶乘结果突变为负数或0是因整数溢出:64位PHP中PHP_INT_MAX为9223372036854775807,21!即超限,导致静默回绕;应前置校验n≤20,改用bcadd/bcmul等字符串化大数运算。

阶乘结果突然变负数或为 0?先查 PHP_INT_MAX 和数据类型
PHP 的整数是有上限的,int 类型在 64 位系统上最大值是 PHP_INT_MAX(通常是 9223372036854775807),而 21! 就已超过这个值。一旦溢出,PHP 不会报错,而是静默回绕成负数或 0——这是阶乘结果“突变”的最常见原因。
实操建议:
立即学习“PHP免费学习笔记(深入)”;
- 用
var_dump(PHP_INT_MAX)确认当前环境整数上限 - 对输入
$n做前置判断:if ($n > 20) { /* 警告或切换算法 */ }(20! = 2432902008176640000,安全边界) - 别依赖
gettype($result) === 'integer'判断是否溢出——溢出后仍是integer类型
用 bcadd/bcmul 做大数阶乘时,参数顺序和字符串化不能漏
PHP 的 BCMath 函数不接受原生整数参与高精度运算,所有操作数必须是字符串,且 bcmul($a, $b) 中 $a 和 $b 顺序错误、或其中一个是整数,会导致结果为 "0" 或计算异常。
实操建议:
立即学习“PHP免费学习笔记(深入)”;
- 每一步乘法前强制转字符串:
$result = bcmul((string)$result, (string)$i) - 初始值必须是字符串:
$result = '1';,不是1 - 别混用
gmp和bc:两者 API 不兼容,gmp_mul返回资源或 GMP 对象,不能直接传给bcmul - 示例片段:
$n = 50;
$result = '1';
for ($i = 2; $i <= $n; $i++) {
$result = bcmul($result, (string)$i);
}
echo $result; // 正确输出 50! 的完整字符串
gmp_fact 最省事,但要注意返回值不是普通整数
gmp_fact($n) 是 PHP 内置的高精度阶乘函数,但它返回的是 GMP 资源(PHP 7.2+ 为 GMP 对象),不能直接 echo 或参与算术运算。直接 echo gmp_fact(30) 可能输出空或乱码,var_dump 显示的是对象结构而非数字。
实操建议:
立即学习“PHP免费学习笔记(深入)”;
- 必须用
gmp_strval()转成字符串才能显示:echo gmp_strval(gmp_fact(30)); - 若需后续计算,保持 GMP 类型,用
gmp_add、gmp_mul等配套函数,别用+或* - 检查扩展是否启用:
extension_loaded('gmp'),某些 Docker 镜像或精简版 PHP 默认不带 GMP
递归写法在大 $n 下栈溢出,比溢出更早崩溃
用递归实现阶乘(如 function fact($n) { return $n )看似简洁,但当 $n > 1000 左右就可能触发 Fatal error: Maximum function nesting level 或直接 segfault,尤其在低内存或 Xdebug 开启时。
实操建议:
立即学习“PHP免费学习笔记(深入)”;
- 一律改用迭代(
for循环),避免调用栈压力 - 若坚持递归,加深度防护:
if ($n > 200) throw new InvalidArgumentException('n too large for recursion'); - 注意 Xdebug 的
xdebug.max_nesting_level默认仅 256,调试时容易误判为逻辑错误
真正难排查的,往往是溢出后数值“看起来还像整数”——比如 25! 在 64 位 PHP 上显示为 -1539835312772124160,没人会第一反应去查 PHP_INT_MAX。动手前先 var_dump($n, $result, gettype($result)),比翻文档快得多。











