c_str()始终返回以'\0'结尾的C风格字符串,data()仅保证前size()字节为内容但不保证末尾有'\0';C++11明确二者行为差异,C++17强化非空终止语义并支持string_view。

data() 返回内部字符数组的指针,不保证末尾有 '\0';c_str() 也返回指针,但强制保证以 '\0' 结尾。
这是最根本的区别:
c_str() 始终返回一个以 '\0' 结尾的 C 风格字符串(null-terminated),可直接传给 printf、strlen、open() 等 C 函数。data() 在 C++11 之前行为未定义(可能不以 '\0' 结尾);C++11 起明确要求 data() 返回的指针指向的前 size() 个字节是字符串内容,但不保证第 size() 个字节是 '\0'。不过实践中多数实现仍会预留空字符(尤其在非 short-string 优化情况下),但你不该依赖它。C++11 是分水岭:
data() 行为 —— 它与 c_str() 返回相同地址,且 data() + size() 处的字节**不一定**是 '\0'(即不要求 null-termination)。但标准允许实现让 data()[size()] == '\0',只是你不能假设它一定成立。std::string_view,其构造函数接受 data(), size() 形式,正体现了对非 null-terminated 字符序列的支持。看调用目标是否需要 '\0':
立即学习“C++免费学习笔记(深入)”;
fopen(filename.c_str(), "r"))、格式化输出(printf("%s", s.c_str()))、系统 API(execv(argv[0], &argv[0]))→ 必须用 c_str()。string_view、或需要访问原始字节(含可能的 '\0' 字符)→ 优先用 data(),并显式传长度:write(fd, s.data(), s.size())。s.data() == s.c_str() 在绝大多数实现中为真,但逻辑上不该用 == 比较它们——比较无意义,且未来实现可能不同(比如 SSO 优化下某次 resize 后 data() 重分配而 c_str() 缓存未更新,虽然标准禁止这种缓存不一致)。下面代码在 C++11+ 中是**未定义行为(UB)**:
std::string s = "hello\0world"; // 含嵌入 '\0'
const char* p = s.data();
printf("%s", p); // ❌ 只打印 "hello",且行为不可靠:%s 遇到第一个 '\0' 就停,但 data() 不保证后续安全正确做法是:
'\0'),不用 %s,改用循环或 std::cout.write(s.data(), s.size());'\0',并用 c_str()。基本上就这些。核心就一条:要 null-terminated → 用 c_str();只要原始字节+长度 → 用 data()。C++11 把这事说清楚了,别再凭经验混用了。
以上就是c++++中的std::basic_string::data()和c_str()的区别_c++ C++11/17字符串标准变化【核心】的详细内容,更多请关注php中文网其它相关文章!
c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号