c – 为什么std :: string_view :: data不包含空终止符?

代码具有未定义的行为:
#include <string_view>
#include <iostream>

using namespace std::string_view_literals;

void foo(std::string_view msg) {
    std::cout << msg.data() << '\n'; // undefined behavior if 'msg' is not null-
                                     // terminated

    // std::cout << msg << '\n'; is not undefined because operator<< uses
    //                           iterators to print 'msg',but that's not the point
}

int main() {
    foo("hello"sv); // not null-terminated - undefined behavior
    foo("foo");     // same,even more dangerous
}

原因是std::string_view可以存储非空终止字符串,并且在调用数据时不包含空终止符.这是非常有限的,为了使上面的代码定义行为,我必须构造一个std :: string:

std::string str{ msg };
std::cout << str.data() << '\n';

在这种情况下,这确实使std :: string_view变得不必要,我仍然要复制传递给foo的字符串,那么为什么不使用移动语义并将msg更改为std :: string?这可能会更快,但我没有衡量.

无论哪种方式,每次我想要将const char *传递给只接受const char *的函数时,必须构造一个std :: string是有点不必要的,但是必须有一个理由让委员会以这种方式决定它.

那么,为什么std::string_view::data不会返回像std :: string :: data这样的以null结尾的字符串?

解决方法

So,why does std::string_view::data not return a null-terminated
string like std::string::data

仅仅因为它不能. string_view可以是更大的字符串(字符串的子字符串)的更窄视图.这意味着查看的字符串不必在特定视图的末尾具有空终止.由于显而易见的原因,您无法将空终止符写入基础字符串,并且无法创建字符串的副本并返回char *而不会发生内存泄漏.

如果您想要一个空终止字符串,则必须从中创建一个std :: string副本.

让我展示一下std :: string_view的好用:

auto tokenize(std::string_view str,Pred is_delim) -> std::vector<std::string_view>

这里生成的向量包含标记作为较大字符串的视图.

相关文章

本程序的编译和运行环境如下(如果有运行方面的问题欢迎在评...
水了一学期的院选修,万万没想到期末考试还有比较硬核的编程...
补充一下,先前文章末尾给出的下载链接的完整代码含有部分C&...
思路如标题所说采用模N取余法,难点是这个除法过程如何实现。...
本篇博客有更新!!!更新后效果图如下: 文章末尾的完整代码...
刚开始学习模块化程序设计时,估计大家都被形参和实参搞迷糊...