0

0

C++ vector越界访问后果 C++缓冲区溢出攻击原理科普【安全】

尼克

尼克

发布时间:2026-01-25 11:54:02

|

898人浏览过

|

来源于php中文网

原创

vector::at()越界抛std::out_of_range异常,operator[]越界导致未定义行为;越界读可能泄露数据,越界写会破坏内存;未触发SIGSEGV不等于安全,因越界可能落在可访问内存页内。

c++ vector越界访问后果 c++缓冲区溢出攻击原理科普【安全】

vector::at() 和 operator[] 的越界行为差异

C++ 标准库 std::vector 对越界访问不做强制检查,但提供两种不同策略:operator[] 完全不检查,at() 会抛出 std::out_of_range 异常。
这导致同一段越界代码在不同写法下表现截然不同:

  • v[i](i ≥ v.size())→ 未定义行为(UB),可能读到垃圾值、崩溃,也可能“恰好”没出事
  • v.at(i)(i ≥ v.size())→ 立即抛异常,便于定位问题

实际开发中,operator[] 常用于性能敏感路径(如内层循环),但一旦索引逻辑有误,错误会隐藏得很深;而 at() 更适合调试期或用户输入驱动的索引场景。

越界读 vs 越界写:危害等级完全不同

越界访问不是“都一样危险”,读和写的后果差异极大:

  • 越界读(如 v[i] 取值,i 超出范围):可能泄露上相邻变量、返回随机整数、触发段错误,但一般不破坏程序控制流
  • 越界写(如 v[i] = x,i 超出范围):直接覆写内存,可能覆盖:
    • 同一栈帧的其他局部变量(改掉标志位、指针地址)
    • 函数返回地址(常见于原始数组,vector 内部缓冲区若在栈上分配则同理)
    • vector 自身的元数据(如 sizecapacity 字段,若内存布局紧凑且编译器未 padding)

尤其注意:即使 vector 数据在堆上分配,其内部指针(_M_start 等)仍在栈/对象内,越界写可能先破坏这些指针,再引发二次崩溃。

为什么 vector 越界不总触发 SIGSEGV?

SIGSEGV 不是“越界就来”,而是“访问了操作系统标记为不可读/不可写的页”。关键点在于:

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

SoftGist
SoftGist

SoftGist是一个软件工具目录站,每天为您带来最好、最令人兴奋的软件新产品。

下载
  • vector 的底层内存通常由 new 分配,连续一页或多页堆内存,相邻地址大概率可读可写
  • 越界偏移较小时(如 +1~+10 个元素),大概率落在同一内存页内 → 不触发信号,只造成静默数据污染
  • 只有当越界落到:
    • 堆边界外的保留区(如 glibc 的 top chunk 后空隙)
    • 其他已映射页的保护区域(如 mprotect 设为 PROT_NONE
    • 栈溢出撞到栈 guard page(vector 在栈上时更易触发)

换句话说:没崩溃 ≠ 没问题,反而更危险——你无法靠 crash 发现它。

缓冲区溢出攻击如何利用 vector 越界?

真实攻击不依赖 vector,但原理相通:攻击者通过可控输入诱导越界写,篡改关键内存。例如:

  • 若某服务将用户 ID 存入 vector ids,又用未经校验的索引执行 ids[user_input] = new_val
  • 攻击者传入极大 user_input(如 0x7fffffffffff),使指针计算后指向返回地址附近
  • 若该 vector 对象与函数返回地址在栈上紧邻(取决于编译器布局、优化等级),一次越界写就可劫持控制流

现代防护(ASLR、stack canary、W^X)大幅增加难度,但 vector 越界仍是内存破坏类漏洞的常见入口点——尤其在禁用异常、关闭 sanitizer 的生产构建中,它几乎不留痕迹。

越界本身不生成 shellcode,但它能打开那扇门;而门后有没有攻击面,取决于你把什么放在了隔壁内存里。

相关专题

更多
string转int
string转int

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

381

2023.08.02

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

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

542

2024.08.29

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

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

53

2025.08.29

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

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

197

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

394

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

574

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

394

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

574

2023.08.10

c++ 根号
c++ 根号

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

41

2026.01.23

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Rust 教程
Rust 教程

共28课时 | 4.8万人学习

PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.7万人学习

Git 教程
Git 教程

共21课时 | 3万人学习

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

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