直接说结论:std::stringstream 在高频、短字符串拼接或数字转换场景下,性能开销远高于预期,主要来自内存分配、locale 依赖和内部缓冲区管理。它不是“慢”,而是“不该用的地方用了就明显拖慢”。
std::stringstream 内部持有 std::stringbuf,默认使用动态分配的缓冲区(即使内容只有几个字
符)。每次新建对象都可能触发 new,析构时又 delete——在循环里写 for (...) { std::stringstream ss; ss 就是典型陷阱。
ss.str(std::string{}) 清空内容(复用缓冲区),但注意这仍可能触发一次小分配(取决于 libstdc++/libc++ 实现)str() 调用后倾向于保留缓冲区;GCC libstdc++ 则更倾向释放——行为不跨平台一致对单个整数或浮点数做格式化输出,ss 需走完整的格式化路径:检查 flags()、查 locale、处理填充宽度、调用 num_put facet——而 std::to_string 是无 locale、无格式控制的裸转换。
int x = 123; // 慢(隐式 locale + 格式逻辑) std::stringstream ss; ss << x; std::string s1 = ss.str(); // 快(直接 memcpy + 简单除法) std::string s2 = std::to_string(x);
std::to_string 不支持进制、精度、填充等格式控制,纯转换场景优先选它std::hex, std::setprecision)时,std::stringstream 才不可替代——但应尽量复用实例ss 比 std::to_string 慢一个数量级
写 ss 看似简洁,但每个 都要检查流状态、调用对应 operator 重载、处理类型擦除开销。而现代 C++ 中,短字符串拼接用 operator+ 或 std::format(C++20)更直接。
// 较慢(流机制冗余)
std::stringstream ss;
ss << "file_" << id << ".log";
std::string name = ss.str();
// 更快(编译期长度可知,SSO 友好)
std::string name = "file_" + std::to_string(id) + ".log";
// C++20 起推荐(零运行时解析、类型安全)
std::string name = std::format("file_{}.log", id);
std::format 编译期检查格式串,运行时无 locale 开销,libc++/MSVC 已优化到接近 + 拼接性能absl::StrCat(Google Abseil)或 fmt::format(比标准库更成熟)ss 中若 s1、s2 是 std::string_view,仍会触发临时 std::string 构造
真正难绕开的,是需要 locale 敏感格式化(货币符号、千分位、宽字符本地化)或运行时动态构建复杂格式串的场景。其余大多数情况,std::stringstream 是“能用”,但不是“该用”。替换前先看清楚:你到底要的是「流式组合能力」,还是仅仅「把几个东西连成一个字符串」。