std::String_view 是 c++17 引入的轻量级非拥有式字符串引用类型,通过指针和长度封装实现高效操作。1. 它避免不必要的内存拷贝,适用于函数参数传递和高频调用场景;2. 占用内存小,适合存储大量字符串引用;3. 使用时需注意不可长期保存指向局部变量的视图、不可修改内容及需手动管理生命周期;4. 推荐用于读多写少的场景,如解析、日志、字符串查找等,提升性能并减少开销。
字符串视图(std::string_view)是 C++17 引入的一个轻量级非拥有式字符串引用类型,它为处理字符串提供了更高效、更安全的方式。在现代 C++ 编程中,合理使用 std::string_view 可以显著提升程序性能,尤其是在频繁传递和操作字符串的场景中。
什么是 std::string_view
std::string_view 是一个不拥有字符串数据的对象,它只是对已有字符串内容的“视图”。它可以指向 std::string、C 风格字符串(const char*),也可以作为函数参数、返回值等使用。
它本质上是一个指针加长度的封装,因此不会进行内存拷贝,也不会修改原始数据。这使得它的构造和传递都非常高效。
立即学习“C++免费学习笔记(深入)”;
举个例子:
std::string str = "hello world"; std::string_view sv(str); // 不拷贝字符串,只是引用
使用 std::string_view 的性能优势
减少不必要的拷贝
传统做法中,如果你要传一个字符串给函数,通常会用 const std::string&。但即使这样,如果传的是字面量 “hello”,编译器可能还是会构造一个临时 std::string 对象。
而用 std::string_view,不管是 std::string 还是 const char*,都可以直接构造,不需要额外拷贝。
比如:
void process(const std::string_view& sv) { // 处理 sv } std::string s = "test"; process(s); // OK process("hello"); // 也 OK,无需构造临时 std::string
更高效的函数接口设计
将函数参数设为 std::string_view 而不是 const std::string&,可以避免不必要的构造和销毁操作。尤其在高频调用的函数中,这种优化效果非常明显。
例如:
- 接口函数用于解析字符串
- 日志打印函数
- 字符串查找/比较等辅助函数
这些地方都应该优先考虑使用 std::string_view。
内存占用小,适合容器存储
由于 std::string_view 只包含指针和长度,大小通常是两个指针的长度(比如 16 字节)。相比 std::string 动辄几十字节甚至更多,它非常适合在容器中存储大量字符串引用。
比如你可以这样写:
std::vector<std::string_view> tokens;
只要确保所引用的原始字符串生命周期足够长即可。
使用时需要注意的问题
虽然 std::string_view 很方便,但也有几个常见“坑”需要注意:
-
不要保存长期使用的 string_view 指向局部字符串
比如函数返回一个指向局部变量的 string_view,会导致悬空引用。 -
不能修改内容
它只是一个只读视图,不能用来修改原始字符串。 -
不能自动管理内存
必须自己保证底层字符串的生命周期。
举个反例:
std::string_view bad() { std::string temp = "temp"; return temp; // 错误:temp 析构后,返回的 string_view 成了悬空引用 }
什么时候应该用 std::string_view
简单来说,只要是“读多写少”的字符串操作,std::string_view 都是首选。
基本上就这些。合理使用 std::string_view 可以让代码更高效、更清晰,同时也能避免很多不必要的拷贝和构造开销。