如何解决为什么 strlen 在特定数字中不起作用
我做了这个功能。看起来它正在工作,但是当涉及到 20 位数字时,返回值为 19。我想知道为什么会发生这个问题..
我的功能
function sumDigits($n) {
return strlen($n);
}
echo sumDigits(100); //3
echo sumDigits(1000); //4
echo sumDigits(12345); //5
echo sumDigits(1000000000); //10
echo sumDigits(145874589632); //12
echo sumDigits(0); //1
echo sumDigits(12345698745254856320); //19 <-- Why not 20?
你能帮我解释一下吗?
非常感谢。
解决方法
首先,我要指出您的函数名称具有误导性,因为您并没有真正对数字的值求和,而是对数字进行计数。所以我会调用你的函数 countDigits
而不是 sumDigits
。
它对大数不起作用的原因是字符串表示将切换到科学记数法,所以你实际上得到的长度是“1.2345698745255E+19”而不是“12345698745254856320”
如果你只对整数感兴趣,用对数会得到更好的结果:
function countDigits($n) {
return ceil(log10($n));
}
对于有小数的数字,没有很好的解决办法,因为64位浮点数的精度是有限的to about 16 significant digits,所以即使你提供更多的数字,尾随小数也会被去掉——这与您的功能无关,而是与数字本身的精度有关。例如,您会发现这两个文字是相等的:
if (1.123456789123456789123456789 == 1.12345678912345678) echo "equal";
,
因为你的函数参数是整数,超出限制。
如果你转储它,它实际上显示以下内容:
1.2345698745255E+19
- 19 个字母。
如果您执行以下操作,它将返回 20 - 注意引号,它将输入声明为字符串。
echo sumDigits("12345698745254856320"); //19 <-- Why not 20? -> now will be 20
根据 documentation,strlen()
需要一个字符串,因此会发生强制转换。使用默认设置,您将获得 1.2345698745255E+19
:
var_dump((string)12345698745254856320);
string(19) "1.2345698745255E+19"
根本问题是 PHP 将您的整数文字转换为浮点数,因为它超过了 PHP_INT_MAX
,因此它不能表示为整数:
var_dump(12345698745254856320,PHP_INT_MAX);
在 64 位 PHP 中:
float(1.2345698745254857E+19)
int(9223372036854775807)
您可以更改显示设置以避免使用 E
表示法,但此时您已经失去了精度。
将整数存储为固定字节数的计算机语言不允许任意精度。最好的机会是切换到字符串:
var_dump('12345698745254856320',strlen('12345698745254856320'));
string(20) "12345698745254856320"
int(20)
... 如果您需要实际的数学运算,还可以选择使用任意精度的库,例如 BCMath 或 GMP。
同样重要的是要考虑到,此类问题有时表明您的输入数据并不是真正的整数,而只是一个非常长的纯数字字符串。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。