0

0

c++ deque容器详解_c++双端队列使用教程

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-03 17:31:47

|

957人浏览过

|

来源于php中文网

原创

c++kquote>deque 是分段连续内存的容器,由多个固定大小缓冲区和中控数组组成;与 vector 的整体连续内存不同,它支持 O(1) 头尾操作但不保证随机访问缓存友好、无 data() 函数、迭代器失效规则更复杂且内存开销更大。

c++ deque容器详解_c++双端队列使用教程

deque 是什么,和 vector 有什么本质区别

deque 不是 vector 的“加强版”,也不是“带头尾插入的 vector”。它的内存布局是分段连续的:由多个固定大小的缓冲区(通常为 512 字节或按元素大小对齐)组成,通过中控数组(map)索引。这决定了它支持 O(1) 头尾插入/删除,但不保证整体内存连续——所以 &v[0] 不能安全转成原生指针数组,std::sort 也不能直接用于整个 deque(会编译失败或行为未定义)。

常见误用场景:

  • deque 当作 vector 用,频繁调用 operator[] 并期望缓存友好 —— 实际跳转开销比 vector 高
  • deque::data() 给 C 接口 —— deque 没有 data() 成员函数(C++11 起只有 vectorstring 有)
  • 假设 push_back 后所有迭代器永不失效 —— 实际上头尾插入只使对应端的迭代器失效,中间迭代器仍有效,这点比 vector 更宽松

哪些操作是真正 O(1),哪些是假的 O(1)

标准明确保证头尾插入/删除、随机访问(operator[]at())、front()/back() 是常数时间。但“常数”不等于“快”:每次 operator[] 需要两次指针运算(查中控数组 + 偏移计算),而 vector 是一次地址加法。

容易被忽略的非 O(1) 操作:

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

  • insert(pos, value)erase(pos) 在中间位置 —— 时间复杂度是 O(min(pos, size() - pos)),因为 deque 会选择从头还是从尾搬运元素
  • resize(n)n > size() 时,可能触发多次缓冲区分配;当 n 时,若缩容跨缓冲区边界,需逐个析构元素
  • assign(first, last) 若范围很大,内部可能反复调用 push_backpush_front,实际性能不如先 clear 再 reserve(但 deque 不支持 reserve)

迭代器失效规则和真实使用建议

deque 迭代器失效比 vector 更复杂,但规律清晰:仅当操作影响到该迭代器所处的缓冲区时才失效。例如:

讯飞绘文
讯飞绘文

讯飞绘文:免费AI写作/AI生成文章

下载
  • push_back():仅使 end() 迭代器失效;其他所有迭代器(包括指向末元素的 --end())保持有效
  • push_front():仅使 begin() 失效;begin()+1 及之后全部有效
  • pop_back():仅使指向原末元素的迭代器失效;end() 自动更新,不额外失效
  • clear():所有迭代器、引用、指针全部失效(因为所有缓冲区都被释放)

实操建议:

  • 避免在循环中边遍历边 erase 中间元素 —— 改用 remove_if + erase(remove_if(...), end()) 惯用法
  • 若需频繁中间插入且在意性能,考虑改用 list(但失去随机访问)或 vector(配合 eraseshrink_to_fit
  • 多线程下,即使只读访问也要注意:size() 不是原子操作,某些实现中可能因缓冲区扩展导致临时重算

内存占用与跨平台兼容性陷阱

deque 的内存开销远大于 vector:除元素本身外,还需中控数组(通常 8~64 个指针)+ 每个缓冲区的管理头(如容量/使用长度)。32 位系统上一个空 deque 可能占 40~80 字节,而 vector 仅 12 字节。

更隐蔽的问题是 allocator 行为差异:

  • libstdc++(GCC)默认使用 __gnu_cxx::__pool_alloc 管理缓冲区,可能复用已释放的缓冲区,导致观察不到内存立即释放
  • libc++(Clang)使用 std::allocator 直接分配,更可预测但无池化优化
  • Windows MSVC 的 deque 实现在 VS2015 前有严重迭代器调试检查开销,开启 _ITERATOR_DEBUG_LEVEL=2 时性能暴跌

如果你看到:

  • valgrind 报告 deque 析构后仍有内存未释放 —— 很可能是分配器池未清空,不是泄漏
  • 不同编译器下 sizeof(deque) 差异很大 —— 这是正常实现差异,不要硬编码偏移
  • std::deque 替代 std::string 做缓冲区 —— 别这么做,string 有 SSO 优化,deque 没有
std::deque d = {1, 2, 3};
d.push_front(0);     // OK: d == {0,1,2,3}
auto it = d.begin(); // it points to 0
d.pop_front();       // it is now invalid
// 使用 it 将导致未定义行为 —— 即使它看起来还能解引用

最常被低估的是缓冲区大小策略:不同标准库实现对“缓冲区长度”的选择逻辑不同(有的按元素大小动态算,有的固定 512 字节),这意味着相同代码在 GCC 和 Clang 下,capacity()(如果 deque 提供该接口,注意:标准并未要求)或实际内存占用可能差数倍。真正在意内存时,别只看 size()

相关专题

更多
string转int
string转int

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

314

2023.08.02

sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

381

2023.09.04

string转int
string转int

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

314

2023.08.02

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

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

524

2024.08.29

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

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

49

2025.08.29

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

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

190

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

996

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

55

2025.10.17

漫画合集pdf网盘入口_漫画解说合集一口气看完
漫画合集pdf网盘入口_漫画解说合集一口气看完

精选高人气漫画合集PDF,一站式网盘入口直达!深度漫画解说整合,一口气看完经典与新作,剧情梳理清晰,省时省力,追漫党必看合集。

3

2026.01.04

热门下载

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

精品课程

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

共48课时 | 6.5万人学习

Excel 教程
Excel 教程

共162课时 | 10.5万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 1.9万人学习

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

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