php中字符串不支持+号直接相加,所谓“字符串加法”实为模拟大整数加法:从右到左逐位相加并处理进位,返回结果字符串;推荐生产环境使用bcadd()函数。

PHP 中字符串本身不支持直接用 + 做加法运算(它会尝试转为数字再相加),所谓“字符串加法”通常指**模拟大整数加法**:将两个表示非负整数的字符串(如 "123" 和 "456")按位相加,返回结果字符串(如 "579"),且能正确处理进位和不同长度。这是经典的手动竖式加法逻辑实现。
核心思路:从右到左逐位模拟手算
把两个字符串看作倒序排列的数字数组,从最低位(末尾)开始,逐位相加并维护进位(carry)。关键点包括:
- 用
strlen()获取长度,通过索引$i = strlen($a) - 1从末尾开始遍历 - 每次取当前位字符(如
$a[$i]),用(int)或ord($c) - ord('0')转为数字 - 累加:当前位和 + 进位,新位值为
$sum % 10,进位为floor($sum / 10) - 任一字符串未处理完,或进位不为 0,就继续循环
- 结果用字符串拼接(推荐
$result = $digit . $result,避免反转)
基础实现(支持非负整数字符串)
以下是一个简洁、可读性强的实现:
function stringAdd(string $a, string $b): string {
$i = strlen($a) - 1;
$j = strlen($b) - 1;
$carry = 0;
$result = '';
<pre class='brush:php;toolbar:false;'>while ($i >= 0 || $j >= 0 || $carry > 0) {
$numA = $i >= 0 ? (int)$a[$i] : 0;
$numB = $j >= 0 ? (int)$b[$j] : 0;
$sum = $numA + $numB + $carry;
$digit = $sum % 10;
$carry = (int)($sum / 10);
$result = (string)$digit . $result;
$i--;
$j--;
}
return $result;} // 示例:echo stringAdd("999", "1"); // 输出 "1000"
增强版:支持前置零与负号(简单扩展)
若需处理带符号或含前导零的输入,可提前归一化:
C编写,实现字符串摘要、文件摘要两个功能。里面主要包含3个文件: Md5.cpp、Md5.h、Main.cpp。其中Md5.cpp是算法的代码,里的代码大多是从 rfc-1321 里copy过来的;Main.cpp是主程序。
立即学习“PHP免费学习笔记(深入)”;
- 用
ltrim($str, '0')去除前导零;若结果为空,视为"0" - 检查是否以
'-'开头:若两数同号,调用上述函数后补符号;若异号,需转为减法(更复杂,建议用 BCMath 或 GMP 扩展) - 注意:纯字符串加法一般默认处理非负整数;负数逻辑涉及借位与符号判断,不建议手动实现,优先使用
bcmul()/bcadd()
为什么不用内置函数?什么时候该用它们?
PHP 的 bcadd($a, $b, $scale) 是专为此类场景设计的:支持任意精度、自动处理符号、前导零和小数。例如:
echo bcadd("12345678901234567890", "98765432109876543210");
// 直接输出 "111111111011111111100"
除非明确要求手写算法(如面试、学习原理),否则生产环境强烈推荐 bcadd()——它基于 GNU BC 库,稳定、高效、无溢出风险。仅当无法启用 bcmath 扩展时,才需手动实现。










