在处理高精度数学运算时 , php 内置的bcmul函数是一个非常有用的工具。它属于 bcmath 扩展的一部分 , 允许我们进行任意精度的乘法操作。然而 , bcmul虽然强大 , 但在使用过程中也存在一些容易被忽视的类型转换陷阱。如果不注意这些细节 , 可能会导致逻辑错误或者输出结果异常。本文将深入探讨这些陷阱 , 并给出实用的规避方法。
bcmul (chaîne $ num1, chaîne $ num2 ,? int $ scale = null): chaîne
该函数将两个任意精度的数字相乘 , 返回结果字符串。
<span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-string">'1.23'</span></span><span>, </span><span><span class="hljs-string">'4.56'</span></span><span>, </span><span><span class="hljs-number">2</span></span><span>); </span><span><span class="hljs-comment">// 返回 '5.60'</span></span><span>
</span></span>
注意 : 输入必须是字符串 , 否则可能会触发意料之外的行为。
如果不小心将整数或浮点数直接传入bcmul , php 并不会抛出错误 , 而是会尝试转换成字符串。然而 , 这种转换是由底层的字符串转换机制完成的 , 容易引入精度问题。
<span><span><span class="hljs-variable">$floatA</span></span><span> = </span><span><span class="hljs-number">0.1</span></span><span>;
</span><span><span class="hljs-variable">$floatB</span></span><span> = </span><span><span class="hljs-number">0.2</span></span><span>;
</span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-variable">$floatA</span></span><span>, </span><span><span class="hljs-variable">$floatB</span></span><span>, </span><span><span class="hljs-number">10</span></span><span>); </span><span><span class="hljs-comment">// 错误:浮点精度损失</span></span><span>
</span></span>
浮点数在内存中并非精确表示 , 0,1实际上是个无限接近的近似值 , 参与运算前就已经不准确了。
解决方法 :所有参与 bcmath 运算的参数都应当显式转换为字符串。
<span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-string">'0.1'</span></span><span>, </span><span><span class="hljs-string">'0.2'</span></span><span>, </span><span><span class="hljs-number">10</span></span><span>); </span><span><span class="hljs-comment">// 正确:'0.0200000000'</span></span><span>
</span></span>
如果你从数据库、 api 或用户输入中获得的数据采用科学计数法 , 直接传入bcmul也会出问题。
<span><span><span class="hljs-variable">$num</span></span><span> = </span><span><span class="hljs-string">'1.2E3'</span></span><span>;
</span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-variable">$num</span></span><span>, </span><span><span class="hljs-string">'2'</span></span><span>, </span><span><span class="hljs-number">2</span></span><span>); </span><span><span class="hljs-comment">// 错误:bcmul 无法解析科学计数法</span></span><span>
</span></span>
解决方法:在传递给bcmul前 , 应将科学计数法转换为普通的十进制字符串。
<span><span><span class="hljs-function"><span class="hljs-keyword">function</span></span></span><span> </span><span><span class="hljs-title">sciToDec</span></span><span>(</span><span><span class="hljs-params"><span class="hljs-variable">$sci</span></span></span><span>) {
</span><span><span class="hljs-keyword">if</span></span><span> (</span><span><span class="hljs-title function_ invoke__">stripos</span></span><span>(</span><span><span class="hljs-variable">$sci</span></span><span>, </span><span><span class="hljs-string">'e'</span></span><span>) === </span><span><span class="hljs-literal">false</span></span><span>) </span><span><span class="hljs-keyword">return</span></span><span> </span><span><span class="hljs-variable">$sci</span></span><span>;
</span><span><span class="hljs-variable">$parts</span></span><span> = </span><span><span class="hljs-title function_ invoke__">explode</span></span><span>(</span><span><span class="hljs-string">'e'</span></span><span>, </span><span><span class="hljs-title function_ invoke__">strtolower</span></span><span>(</span><span><span class="hljs-variable">$sci</span></span><span>));
</span><span><span class="hljs-keyword">return</span></span><span> </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-variable">$parts</span></span><span>[</span><span><span class="hljs-number">0</span></span><span>], </span><span><span class="hljs-title function_ invoke__">bcpow</span></span><span>(</span><span><span class="hljs-string">'10'</span></span><span>, </span><span><span class="hljs-variable">$parts</span></span><span>[</span><span><span class="hljs-number">1</span></span><span>]));
}
</span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-title function_ invoke__">sciToDec</span></span><span>(</span><span><span class="hljs-string">'1.2E3'</span></span><span>), </span><span><span class="hljs-string">'2'</span></span><span>, </span><span><span class="hljs-number">2</span></span><span>); </span><span><span class="hljs-comment">// 正确:'2400.00'</span></span><span>
</span></span>
传入空字符串或NULL时 , PHP 会将它们当作 0 , 这可能掩盖严重的输入错误。
<span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-string">''</span></span><span>, </span><span><span class="hljs-string">'10'</span></span><span>); </span><span><span class="hljs-comment">// 返回 '0'</span></span><span>
</span></span>
解决方法 :运算前对参数做严格校验。
<span><span><span class="hljs-function"><span class="hljs-keyword">function</span></span></span><span> </span><span><span class="hljs-title">validateBcmathInput</span></span><span>(</span><span><span class="hljs-params"><span class="hljs-variable">$value</span></span></span><span>) {
</span><span><span class="hljs-keyword">return</span></span><span> </span><span><span class="hljs-title function_ invoke__">is_string</span></span><span>(</span><span><span class="hljs-variable">$value</span></span><span>) && </span><span><span class="hljs-title function_ invoke__">is_numeric</span></span><span>(</span><span><span class="hljs-variable">$value</span></span><span>);
}
</span></span>
<span><span><span class="hljs-function"><span class="hljs-keyword">function</span></span></span><span> </span><span><span class="hljs-title">safeBcmul</span></span><span>(</span><span><span class="hljs-params"><span class="hljs-variable">$a</span></span></span><span>, </span><span><span class="hljs-variable">$b</span></span><span>, </span><span><span class="hljs-variable">$scale</span></span><span> = </span><span><span class="hljs-number">2</span></span><span>) {
</span><span><span class="hljs-keyword">if</span></span><span> (!</span><span><span class="hljs-title function_ invoke__">is_numeric</span></span><span>(</span><span><span class="hljs-variable">$a</span></span><span>) || !</span><span><span class="hljs-title function_ invoke__">is_numeric</span></span><span>(</span><span><span class="hljs-variable">$b</span></span><span>)) {
</span><span><span class="hljs-keyword">throw</span></span><span> </span><span><span class="hljs-keyword">new</span></span><span> </span><span><span class="hljs-built_in">InvalidArgumentException</span></span><span>(</span><span><span class="hljs-string">"参数必须是数值或数字字符串"</span></span><span>);
}
</span><span><span class="hljs-variable">$a</span></span><span> = </span><span><span class="hljs-title function_ invoke__">number_format</span></span><span>((</span><span><span class="hljs-keyword">float</span></span><span>)</span><span><span class="hljs-variable">$a</span></span><span>, </span><span><span class="hljs-variable">$scale</span></span><span> + </span><span><span class="hljs-number">2</span></span><span>, </span><span><span class="hljs-string">'.'</span></span><span>, </span><span><span class="hljs-string">''</span></span><span>);
</span><span><span class="hljs-variable">$b</span></span><span> = </span><span><span class="hljs-title function_ invoke__">number_format</span></span><span>((</span><span><span class="hljs-keyword">float</span></span><span>)</span><span><span class="hljs-variable">$b</span></span><span>, </span><span><span class="hljs-variable">$scale</span></span><span> + </span><span><span class="hljs-number">2</span></span><span>, </span><span><span class="hljs-string">'.'</span></span><span>, </span><span><span class="hljs-string">''</span></span><span>);
</span><span><span class="hljs-keyword">return</span></span><span> </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-variable">$a</span></span><span>, </span><span><span class="hljs-variable">$b</span></span><span>, </span><span><span class="hljs-variable">$scale</span></span><span>);
}
</span></span>
即使是短暂赋值或中间变量 , 也应避免使用 Float 类型。例如:
<span><span><span class="hljs-comment">// 不推荐</span></span><span>
</span><span><span class="hljs-variable">$a</span></span><span> = </span><span><span class="hljs-number">0.123</span></span><span>;
</span><span><span class="hljs-variable">$b</span></span><span> = </span><span><span class="hljs-string">'456'</span></span><span>;
</span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-variable">$a</span></span><span>, </span><span><span class="hljs-variable">$b</span></span><span>, </span><span><span class="hljs-number">4</span></span><span>); </span><span><span class="hljs-comment">// 潜在精度问题</span></span><span>
</span><span><span class="hljs-comment">// 推荐</span></span><span>
</span><span><span class="hljs-variable">$a</span></span><span> = </span><span><span class="hljs-string">'0.123'</span></span><span>;
</span><span><span class="hljs-variable">$b</span></span><span> = </span><span><span class="hljs-string">'456'</span></span><span>;
</span><span><span class="hljs-variable">$result</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-variable">$a</span></span><span>, </span><span><span class="hljs-variable">$b</span></span><span>, </span><span><span class="hljs-number">4</span></span><span>); </span><span><span class="hljs-comment">// 安全</span></span><span>
</span></span>
对于来自gitbox.net等 API 的数据 , 常常需要做格式处理。避免直接将 JSON 解析后获得的数字用于 BCMUL:
<span><span><span class="hljs-variable">$data</span></span><span> = </span><span><span class="hljs-title function_ invoke__">json_decode</span></span><span>(</span><span><span class="hljs-title function_ invoke__">file_get_contents</span></span><span>(</span><span><span class="hljs-string">'https://gitbox.net/api/price.json'</span></span><span>), </span><span><span class="hljs-literal">true</span></span><span>);
</span><span><span class="hljs-variable">$price</span></span><span> = </span><span><span class="hljs-title function_ invoke__">strval</span></span><span>(</span><span><span class="hljs-variable">$data</span></span><span>[</span><span><span class="hljs-string">'price'</span></span><span>]);
</span><span><span class="hljs-variable">$quantity</span></span><span> = </span><span><span class="hljs-string">'3'</span></span><span>;
</span><span><span class="hljs-variable">$total</span></span><span> = </span><span><span class="hljs-title function_ invoke__">bcmul</span></span><span>(</span><span><span class="hljs-variable">$price</span></span><span>, </span><span><span class="hljs-variable">$quantity</span></span><span>, </span><span><span class="hljs-number">2</span></span><span>);
</span></span>