0

0

c++中如何使用std::span安全操作子数组_c++20现代编程【实例】

尼克

尼克

发布时间:2026-01-17 16:46:02

|

539人浏览过

|

来源于php中文网

原创

std::span是C++20引入的非拥有式连续内存视图,不能当普通指针用,因其不管理生命周期,底层容器销毁后访问即UB;构造需确保源对象生命周期覆盖span使用期,subspan无运行时检查,传参需注意const与模板推导。

c++中如何使用std::span安全操作子数组_c++20现代编程【实例】

std::span 是什么,为什么不能当普通指针用

std::span 是 C++20 引入的轻量视图类型,它不拥有数据,只持有指向连续内存的指针和长度。它不是智能指针,也不是容器,更不是 std::vector 的替代品——它没有分配、释放、扩容能力。常见误用是把它当作“安全版裸指针”传给旧接口后随意修改底层数据,结果引发未定义行为(UB),尤其在原容器生命周期结束之后仍访问 std::span

关键约束:只要底层容器(如 std::vector数组)还活着,且没被移动或销毁,std::span 就安全;一旦容器析构,std::span 立即变成悬空视图。

如何从 std::vector 或原生数组构造合法 span

构造本身很简单,但合法性取决于源对象的生命周期是否覆盖 span 的使用期。以下是最常用且安全的场景:

  • 从局部 std::vector 构造 std::span,并在同一作用域内使用(推荐)
  • 从函数参数传入的 const std::vector&T* + size_t 构造(需确保调用方保证生命周期)
  • 从栈数组(如 int arr[10])构造,不跨函数返回

反例:不要返回局部 std::vectorstd::span,也不要保存由临时 std::vector{...} 构造的 std::span

立即学习C++免费学习笔记(深入)”;

std::vector data = {1, 2, 3, 4, 5};
std::span view = data; // OK:view 引用 data 的堆内存
std::span cview = data; // OK:隐式转换为 const 版本
int arr[] = {10, 20, 30};
std::span stack_view{arr}; // OK:编译期知道大小,更安全

切分子数组:subspan() 的边界检查与陷阱

subspan() 是获取子视图的核心方法,但它不做运行时越界检查(C++20 标准明确要求它是 noexcept 且零开销)。如果起始位置或长度超出当前 span 范围,行为是未定义的——不会抛异常,也不会断言,只会静默出错。

Text-To-Song
Text-To-Song

免费的实时语音转换器和调制器

下载

安全做法:手动校验参数,或封装一层带检查的辅助函数。尤其注意:subspan(pos)subspan(pos, count) 中的 pos 是索引,不是指针偏移;count 为 0 是合法的(返回空 span)。

  • view.subspan(2, 3):从索引 2 开始取 3 个元素 → {3,4,5}(若原 view 长度 ≥5)
  • view.subspan(10):若 view.size() ,UB —— 不要依赖调试器或 ASan 自动捕获
  • view.subspan(view.size()):合法,返回空 span(data()==nullptrsize()==0
std::vector buf(100);
std::span full{buf};
if (offset < full.size() && len <= full.size() - offset) {
    std::span sub = full.subspan(offset, len); // 此时才安全
}

传递 span 给函数时要注意 const 与模板推导

函数参数用 std::span 还是 std::span,直接影响能否修改数据,也影响模板实参推导。常见错误是写成 void f(std::span s) 却传入 const std::vector&,导致编译失败(因为 const vectordata() 返回 const int*,无法隐式转为 int*)。

更健壮的写法是使用模板参数自动推导,或统一用 const 视图接收只读需求:

  • 只读处理:优先用 std::span 参数
  • 需要修改:确保传入的是非 const 容器引用,或明确要求调用方提供可写视图
  • 泛型函数:用 template void f(std::span s),让编译器根据实参决定 T
void process_readonly(std::span s) {
    for (double x : s) { /* ... */ }
}
void process_mutable(std::span s) {
    for (float& x : s) x *= 2.f;
}
// 调用:
std::vector d = {1.1, 2.2};
process_readonly(d); // OK
std::vector f = {1.f, 2.f};
process_mutable(f); // OK
// process_mutable(d); // 编译错误:double* 无法转 float*

最易被忽略的一点:std::span 的 lifetime 完全独立于其构造来源,它不参与 RAII,也不触发任何检查。你得自己画清楚变量作用域和所有权链——这恰恰是它高效的前提,也是它危险的根源。

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

197

2023.11.20

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

524

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

97

2025.11.27

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号