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”
增强版:支持前置零与负号(简单扩展)
若需处理带符号或含前导零的输入,可提前归一化:
立即学习“PHP免费学习笔记(深入)”;
- 用
ltrim($str, '0')去除前导零;若结果为空,视为"0" - 检查是否以
'-'开头:若两数同号,调用上述函数后补符号;若异号,需转为减法(更复杂,建议用 BCMath 或 GMP 扩展) - 注意:纯字符串加法一般默认处理非负整数;负数逻辑涉及借位与符号判断,不建议手动实现,优先使用
bcmul()/bcadd()
为什么不用内置函数?什么时候该用它们?
PHP 的 bcadd($a, $b, $scale) 是专为此类场景设计的:支持任意精度、自动处理符号、前导零和小数。例如:
echo bcadd("12345678901234567890", "98765432109876543210"); // 直接输出 "111111111011111111100"
除非明确要求手写算法(如面试、学习原理),否则生产环境强烈推荐 bcadd()——它基于 gnu BC 库,稳定、高效、无溢出风险。仅当无法启用 bcmath 扩展时,才需手动实现。