
用 % 和 / 取个位、十位、百位最直接
别绕弯子,C++ 里提取 int 各数位,核心就靠取模 % 和整除 /。比如 n = 1234,n % 10 是个位(4),(n / 10) % 10 是十位(3),(n / 100) % 10 是百位(2)——顺序是从右往左剥。
常见错误是写成 n % 100 想拿“后两位”,结果和“十位”混淆;或者对负数不处理,导致 -123 % 10 在 C++ 里是 -3(不是 7),直接崩逻辑。
- 正数场景下放心用:
digit = (n / pow(10, i)) % 10,但pow返回double,强制转int有精度风险,不如手写除法 - 负数必须先取绝对值:
abs(n),否则%行为依赖编译器实现(C++98/C++11 后统一为向零截断,但个位值仍为负) - 注意
0:单独处理,否则循环提取会漏掉末尾的0(比如100,两次/10后变1,再除就没了)
while 循环逐位提取时怎么避免死循环
用 while (n != 0) 提取所有非零数位很常见,但容易卡在 n = -2147483648(INT_MIN)上——因为 abs(INT_MIN) 溢出,且 n = n / 10 对它反复除仍是自身(有符号溢出未定义行为)。
- 安全做法:用
long long临时存abs(n),再进循环 - 循环体里别只写
n /= 10,要配合n % 10一起用,否则位数顺序反了(你得从低位往高位存,再反转,或用栈) - 如果只要某一位(比如第
k位,从右往左数,0 起始),直接算:(abs(n) / static_cast<int>(pow(10, k))) % 10</int>,但pow不推荐;更稳的是预计算幂表或用循环除 10 共k次
std::to_string + 下标访问是不是更简单
是,但代价明确:生成字符串对象,堆分配(小字符串可能 SSO,但不可控),还多一次内存拷贝。纯数值计算场景下,这属于「杀鸡用牛刀」。
立即学习“C++免费学习笔记(深入)”;
典型误用是:为了取个位,写 std::to_string(n).back() - '0',看着短,但比 n % 10 慢一个数量级,且无法处理负号逻辑("-123"[3] 是 '3',但 "-123".back() 是 '3',看似对,实则掩盖了符号处理缺失)。
- 仅推荐用于调试输出、或后续真要按字符处理(比如找某数字出现次数)
- 注意
std::to_string对INT_MIN安全,不会溢出,这点比手算abs稳 - 如果已用
std::string接口,别再转回 int 做算术——那就全程走字符串流也行,但别混用
提取过程中容易被忽略的边界:前导零和位数长度
“提取各数位”听起来只是拆数字,但实际需求常隐含位数对齐,比如把 7 当作 007 处理。C++ 原生整数不存前导零,这部分必须由你补。
- 想固定取 3 位?不能只靠
n % 1000——那是取后三位,7变成7,不是007;得自己格式化输出,或用数组/容器存够 3 个位置再填 - 求位数长度别用
log10(n)+1:对n=0失效(log10(0)未定义),对负数也不适用;稳妥是转字符串取.length(),或循环除 10 计数(记得特判0) - 如果目标是拼接新数字(比如交换十位和个位),直接算数值比拼字符串快,也避免
stoi异常开销
真正麻烦的从来不是“怎么拿到每一位”,而是“拿到之后怎么对齐、怎么保符号、怎么防溢出”。这些点没卡准,一行 % 10 就能让你在测试用例 -2147483648 或 0 上卡半天。










