std::stoi最常用但易抛异常,需容错封装;std::from_chars是c++17最快无异常方案,但不处理空格和进制前缀;atoi静默失败,strtol更可控;手写解析性能最优但缺乏通用性。

std::stoi 用得最多,但会抛异常
绝大多数人直接用 std::stoi 转 std::string 到 int,写法简单:int x = std::stoi("123");。但它在遇到非法字符(如 "123abc")、空字符串或超出 int 范围时,会抛出 std::invalid_argument 或 std::out_of_range 异常。
如果你的输入不可控(比如用户输入、日志解析),不加 try/catch 就 crash。实际项目中建议至少包一层容错:
int safe_stoi(const std::string& s, int default_val = 0) {
try {
size_t pos;
int val = std::stoi(s, &pos);
if (pos != s.length()) return default_val; // 后缀非数字,如 "42x"
return val;
} catch (...) { return default_val; }
}std::from_chars 是 C++17 最快且无异常的方案
std::from_chars 是目前 C++ 标准库中性能最高、最轻量的字符串转数字接口,不分配内存、不抛异常、不依赖 locale。适用于对性能敏感或嵌入式场景。
它返回一个 std::from_chars_result 结构体,含 ptr(解析结束位置)和 ec(错误码):
立即学习“C++免费学习笔记(深入)”;
-
ec == std::errc{}表示成功 -
ec == std::errc::invalid_argument表示无有效数字 -
ec == std::errc::result_out_of_range表示溢出
示例:
std::string s = " -42 ";
int val;
auto [ptr, ec] = std::from_chars(s.data(), s.data() + s.size(), val);
if (ec == std::errc{} && ptr > s.data()) {
// 成功,注意:不会跳过前导空格,s = " -42 " 会失败
// 需手动 trim 或用 data()+offset 调整起始位置
}⚠️ 注意:std::from_chars 不处理前导空格和符号校验("-42" 可以,但 " -42" 不行),也不支持十六进制前缀("0x1A" 需自己截断)。
atoi 和 strtol:C 风格,快但难控制精度
atoi 最简单:int x = atoi(s.c_str());,但它对错误完全静默——非法输入返回 0,无法区分 "0" 和 "abc"。基本只适合“我确定这串一定合法”的内部脚本场景。
strtol 更靠谱,能返回解析终点、识别进制、报告溢出:
-
char* end;接收解析结束地址,可检查是否全解析(end == s.c_str() + s.size()) - 第三个参数指定进制,
0表示自动识别"0x"/"0" - 返回
LONG_MIN/LONG_MAX并设errno表示溢出,需手动判断
示例:
std::string s = "12345";
char* end;
long tmp = std::strtol(s.c_str(), &end, 10);
if (end == s.c_str() + s.size() && tmp >= INT_MIN && tmp <= INT_MAX) {
int x = static_cast<int>(tmp);
}自定义解析(如只处理纯数字)可能比标准库还快
如果你的输入格式高度受限(比如确定是正整数、长度 ≤ 8、无空格),手写解析往往更快,且完全可控:
int parse_positive_int(const std::string& s) {
if (s.empty()) return -1;
int res = 0;
for (char c : s) {
if (c < '0' || c > '9') return -1; // 非数字立即失败
if (res > (INT_MAX - (c-'0')) / 10) return -1; // 溢出预检
res = res * 10 + (c - '0');
}
return res;
}这种写法避免了函数调用开销、异常机制、locale 查表等,LLVM/Clang 下常被完全内联。但代价是失去通用性——负号、空格、进制、科学计数法都得自己补。
真正容易被忽略的是:所有方法对 Unicode、全角数字、BOM 头都无感;如果输入来自文件或网络,务必确认编码是纯 ASCII/UTF-8 且无隐藏控制字符。










