std::span是C++20引入的非拥有型连续内存视图,用于安全高效地替代T*+size_t组合;它可从数组、容器或指针创建,支持边界检查访问、子视图切片(如subspan、first、last)和范围遍历,推荐作为函数参数传递以提升类型安全与代码清晰度,但不管理所指数据生命周期,禁止绑定临时initializer_list或返回局部数组的span。
在 C++20 中,std::span 提供了一种安全且高效的方式来操作连续内存区域,而无需拥有其数据。它是一个“非拥有型”(non-owning)的视图,适用于数组、std::array、std::vector 等连续存储结构。相比原始指针和长度参数组合,std::span 更加类型安全、边界清晰,并能减少出错几率。
std::span 是一个轻量级的包装器,表示一段连续的内存范围。它不管理内存生命周期,只提供对已有数据的安全访问接口。 它可以替代 T* + size_t 这样的参数传递方式,让函数签名更清晰。
常见用途包括:
要使用 std::span,需包含头文件 (C++20 起),并确保编译器支持 C++20 标准。
基本构造方式如下:
int arr[] = {1, 2, 3, 4, 5};
std::span s{arr}; // 自动推导长度为 5
std::vector vec = {10, 20, 30};
std::span s{vec};
int* ptr = ...;
size_t len = ...;
std::span s{ptr, len};
std::span 支持多种安全访问方式:
std::span s = {1, 2, 3};
fo
r (const auto& elem : s) {
std::cout << elem << ' ';
}
// 或使用算法
std::sort(s); // 如果 span 非 const
通过 .subspan() 可以获取原 span 的一部分,形成新的视图,避免复制数据。
std::span s = {0, 1, 2, 3, 4, 5};
auto part = s.subspan(2, 3); // 从索引 2 开始取 3 个元素 → {2,3,4}
也提供便捷方法:
将 std::span 用于函数参数能显著提升代码安全性与可读性。
void process_data(std::spandata) { for (int x : data) { // 处理逻辑 } }
调用时自动适配:
int arr[] = {1,2,3};
std::vector vec = {4,5,6};
process_data(arr); // OK
process_data(vec); // OK
process_data({7,8,9}); // OK,临时数组
建议优先使用 const std::span
尽管 std::span 很强大,但仍需注意以下几点:
示例错误用法:
std::spanbad_function() { int local_arr[10]; return std::span{local_arr}; // 危险!返回指向已销毁数组的视图 }
基本上就这些。std::span 让你以现代 C++ 的方式安全操作连续内存,减少错误,提升表达力。只要记住“非拥有”这一核心原则,就能有效利用它。