跳到主要内容C++容器:stack、queue、deque 基本用法详解 | 极客日志C++算法
C++容器:stack、queue、deque 基本用法详解
C++ STL 标准模板库中的 stack、queue 和 deque 是高频使用的线性容器。deque 是原生序列容器,支持双端高效操作和随机访问;stack 和 queue 是基于 deque 实现的容器适配器,分别遵循后进先出和先进先出规则。文章详细讲解了它们的头文件引入、构造函数、核心成员函数、常见误区以及括号匹配、消息队列、滑动窗口等实战案例,帮助开发者根据场景选择合适的容器。
链路追踪6.2K 浏览 在 C++ STL 标准模板库中,stack(栈)、queue(队列)、deque(双端队列)是三组高频使用的线性容器,其中 stack 和 queue 并非「原生序列容器」,而是基于实现的特殊数据结构,且它们的默认底层容器正是 deque。这三者各有独特的访问规则和适用场景,本文将从核心特性出发,逐步讲解它们的构造、常用操作及实战用法。
容器适配器
一、先理清三者的核心关系与基础认知
在正式学习用法前,先建立对这三个容器的整体认知,避免混淆核心概念:
- deque(双端队列):原生序列容器,底层是「分段连续内存」(并非完全连续,却能模拟连续内存的效果),支持双端高效插入/删除,也支持随机访问,是 stack 和 queue 的默认底层支撑容器。
- stack(栈):容器适配器,基于「后进先出(LIFO, Last In First Out)」规则工作,仅允许在容器的一端进行插入和删除操作,默认以 deque 为底层容器。
- queue(队列):容器适配器,基于「先进先出(FIFO, First In First Out)」规则工作,仅允许在容器的一端插入、另一端删除,默认同样以 deque 为底层容器。
- deque 像一个「两端都能进出的管道」;
- stack 像一个「只有顶端能进出的瓶子」(后放进去的东西先拿出来);
- queue 像一个「排队买票的队伍」(先排队的人先买票,后排队的人后买票)。
二、基础使用前提:头文件与命名空间
使用这三个容器前,必须包含对应的头文件,若想简化代码,可引入 std 命名空间(否则需显式使用 std:: 前缀):
#include <deque>
#include <stack>
#include <queue>
using namespace std;
三、deque(双端队列):双端高效操作的序列容器
deqeue 是「double-ended queue」的缩写,作为原生序列容器,它弥补了 vector(仅尾部高效)和 list(不支持随机访问)的部分短板,具备独特的优势。
3.1. deque 的核心特性
- 底层:分段连续内存,通过中控器维护各段内存的指针,兼顾连续内存的访问效率和链表的内存灵活性。
- 插入/删除:两端(头部、尾部)插入/删除效率极高,时间复杂度 O(1);中间插入/删除效率较低,时间复杂度 O(n)(需移动元素)。
- 访问:支持随机访问(通过下标
[] 或 at()),时间复杂度 O(1),区别于 list 的双向迭代器。
- 迭代器:支持随机访问迭代器(可进行
++、--、+、- 操作,如 it+3),可正向、反向遍历。
- 适用场景:需要频繁在两端操作元素,且偶尔需要随机访问的场景(如滑动窗口、缓存队列)。
3.2. deque 的常用构造函数(4 种高频方式)
和 list、vector 的构造方式高度相似,初学者可快速迁移知识:
deque<int> dq;
deque<int> dq1(5, 10);
deque<int> dq2(3);
vector<int> vec = {1,2,3,4,5};
deque<int> dq3(vec.begin(), vec.end());
deque<int> dq4(vec.begin(), vec.begin()+3);
deque<int> dq5(5, 10);
deque<int> dq6(dq5);
3.3. deque 的常用成员函数(核心操作)
deqeue 的成员函数兼具 vector 和 list 的部分特性,核心操作可分为「增、删、查、改、容器管理」五类:
(1)元素插入(双端 + 中间)
deque<int> dq;
dq.push_back(10);
dq.push_back(20);
dq.push_front(5);
dq.push_front(1);
deque<int>::iterator it = ++dq.begin();
dq.insert(it, 15);
(2)元素删除(双端 + 中间)
deque<int> dq = {1,15,5,10,20};
dq.pop_back();
dq.pop_front();
it = dq.begin();
it = dq.erase(it);
dq.clear();
(3)元素访问与修改(随机访问 + 双端访问)
deqeue 支持两种访问方式,满足不同场景需求:
deque<int> dq = {1,2,3,4,5};
cout << "第 3 个元素(下标 2):" << dq[2] << endl;
cout << "第 4 个元素(下标 3):" << dq.at(3) << endl;
cout << "头部元素:" << dq.front() << endl;
cout << "尾部元素:" << dq.back() << endl;
dq[2] = 30;
dq.front() = 10;
dq.back() = 50;
(4)容器大小与状态管理
deque<int> dq = {1,2,3};
cout << "元素个数:" << dq.size() << endl;
cout << "是否为空:" << dq.empty() << endl;
dq.resize(5, 100);
dq.resize(2);
(5)迭代器遍历示例
#include <deque>
#include <iostream>
using namespace std;
int main() {
deque<int> dq = {10,20,30,40,50};
cout << "正向遍历:";
for (deque<int>::iterator it = dq.begin(); it != dq.end(); it++) {
cout << *it << " ";
}
cout << endl;
cout << "反向遍历:";
for (deque<int>::reverse_iterator it = dq.rbegin(); it != dq.rend(); it++) {
cout << *it << " ";
}
cout << endl;
cout << "下标遍历:";
for (int i = 0; i < dq.size(); i++) {
cout << dq[i] << " ";
}
cout << endl;
return 0;
}
四、stack(栈):后进先出的容器适配器
stack 是一种「适配器容器」,它不直接实现数据存储,而是封装了底层容器(默认 deque),并提供专属的接口,强制限制只能在「栈顶」进行操作,符合后进先出的规则。
4.1. stack 的核心特性
- 访问规则:仅允许在栈顶(容器的一端)进行插入、删除和访问操作,无法访问栈中间或栈底的元素。
- 底层容器:默认使用 deque,也可指定 vector 或 list 作为底层容器(如
stack<int, vector<int>> st;)。
- 无迭代器:不支持迭代器遍历,无法通过循环遍历所有元素(只能通过弹出栈顶元素的方式逐个获取)。
- 时间复杂度:栈顶的插入(push)、删除(pop)、访问(top)操作均为 O(1)。
- 适用场景:需要后进先出逻辑的场景(如括号匹配、函数调用栈、表达式求值、逆序输出数据)。
4.2. stack 的常用构造函数
stack<int> st;
stack<int> st1;
st1.push(10);
st1.push(20);
stack<int> st2(st1);
4.3. stack 的常用成员函数(核心操作)
stack 的接口非常简洁,仅提供与栈顶相关的操作,无复杂逻辑,初学者容易掌握:
#include <stack>
#include <iostream>
using namespace std;
int main() {
stack<int> st;
st.push(10);
st.push(20);
st.push(30);
cout << "当前栈顶元素:" << st.top() << endl;
st.pop();
cout << "出栈后栈顶元素:" << st.top() << endl;
cout << "栈内元素个数:" << st.size() << endl;
cout << "栈是否为空:" << st.empty() << endl;
while (!st.empty()) {
st.pop();
}
cout << "清空后栈是否为空:" << st.empty() << endl;
return 0;
}
pop() 函数无返回值,仅删除栈顶元素,若需要获取栈顶元素,需先调用 top() 查看,再调用 pop() 删除。stack 没有 clear() 成员函数,清空栈需通过循环调用 pop(),直到 empty() 返回 true。当栈为空时,调用 top() 和 pop() 会触发未定义行为,建议操作前先用 empty() 判断。
五、queue(队列):先进先出的容器适配器
queue 同样是「适配器容器」,默认底层容器为 deque,它强制限制「一端插入、另一端删除」,符合先进先出的规则,类似日常生活中的排队场景。
5.1. queue 的核心特性
- 访问规则:队尾(back)插入元素,队头(front)删除元素,仅能访问队头和队尾元素,无法访问队列中间元素。
- 底层容器:默认使用 deque,也可指定 list 作为底层容器(如
queue<int, list<int>> q;,不推荐 vector,因为 vector 头部删除效率低)。
- 无迭代器:不支持迭代器遍历,无法通过循环直接遍历所有元素(只能通过弹出队头元素的方式逐个获取)。
- 时间复杂度:队尾插入(push)、队头删除(pop)、队头/队尾访问均为 O(1)。
- 适用场景:需要先进先出逻辑的场景(如任务队列、消息队列、排队模拟、广度优先搜索(BFS))。
5.2. queue 的常用构造函数
和 stack 类似,queue 的构造方式也较为简洁:
queue<int> q;
queue<int> q1;
q1.push(10);
q1.push(20);
queue<int> q2(q1);
5.3. queue 的常用成员函数(核心操作)
queue 的接口同样简洁,仅提供与队头、队尾相关的操作,逻辑清晰:
#include <queue>
#include <iostream>
using namespace std;
int main() {
queue<int> q;
q.push(10);
q.push(20);
q.push(30);
cout << "当前队头元素:" << q.front() << endl;
cout << "当前队尾元素:" << q.back() << endl;
q.pop();
cout << "出队后队头元素:" << q.front() << endl;
cout << "队列内元素个数:" << q.size() << endl;
cout << "队列是否为空:" << q.empty() << endl;
while (!q.empty()) {
q.pop();
}
cout << "清空后队列是否为空:" << q.empty() << endl;
return 0;
}
注意:pop() 函数无返回值,仅删除队头元素,若需要获取队头元素,需先调用 front() 查看,再调用 pop() 删除。queue 没有 clear() 成员函数,清空队列需通过循环调用 pop(),直到 empty() 返回 true。当队列为空时,调用 front()、back() 和 pop() 会触发未定义行为,操作前务必用 empty() 判断。
六、常见误区与注意事项
- 混淆「容器适配器」与「原生序列容器」:stack 和 queue 是适配器,不是原生容器,它们封装了底层容器(deque/vector/list),仅暴露有限的接口,不支持迭代器和随机访问;而 deque 是原生序列容器,接口更丰富,支持双端操作和随机访问。
- stack/queue 的遍历问题:两者均无迭代器,无法直接遍历,若需查看所有元素,只能通过「获取头部/栈顶元素→删除元素」的循环方式,且该方式会清空容器内的元素(若需保留数据,可先拷贝一份容器)。
- deque 与 vector、list 的选择:
- 需频繁随机访问、插入/删除多在尾部 → 选择 vector;
- 需频繁在中间插入/删除、无需随机访问 → 选择 list;
- 需频繁在两端插入/删除、偶尔随机访问 → 选择 deque。
- stack/queue 的底层容器替换:stack 可指定 vector/list 为底层容器,queue 可指定 list 为底层容器,但不推荐为 queue 指定 vector(vector 头部删除效率低,违背 queue 的设计初衷)。
七、实战案例:三者的典型应用
案例 1:stack 实现括号匹配(判断字符串中的括号是否成对有效)
#include <stack>
#include <iostream>
#include <string>
using namespace std;
bool isValidBracket(const string& s) {
stack<char> st;
for (char c : s) {
if (c == '(' || c == '[' || c == '{') {
st.push(c);
} else {
if (st.empty()) {
return false;
}
char topChar = st.top();
st.pop();
if ((c == ')' && topChar != '(') || (c == ']' && topChar != '[') || (c == '}' && topChar != '{')) {
return false;
}
}
}
return st.empty();
}
int main() {
string s1 = "()[]{}";
string s2 = "([)]";
string s3 = "{[]}";
cout << s1 << " 是否有效:" << (isValidBracket(s1) ? "是" : "否") << endl;
cout << s2 << " 是否有效:" << (isValidBracket(s2) ? "是" : "否") << endl;
cout << s3 << " 是否有效:" << (isValidBracket(s3) ? "是" : "否") << endl;
return 0;
}
案例 2:queue 实现简单消息队列(先进先出处理消息)
#include <queue>
#include <iostream>
#include <string>
using namespace std;
class MessageQueue {
private:
queue<string> msgQueue;
public:
void sendMsg(const string& msg) {
msgQueue.push(msg);
cout << "发送消息:" << msg << endl;
}
void recvMsg() {
if (msgQueue.empty()) {
cout << "无未读消息!" << endl;
return;
}
string msg = msgQueue.front();
msgQueue.pop();
cout << "接收消息:" << msg << endl;
}
int getUnreadCount() const {
return msgQueue.size();
}
};
int main() {
MessageQueue mq;
mq.sendMsg("Hello, C++ STL!");
mq.sendMsg("这是一个队列消息");
mq.sendMsg("消息处理完成~");
cout << "未读消息数量:" << mq.getUnreadCount() << endl;
mq.recvMsg();
mq.recvMsg();
mq.recvMsg();
mq.recvMsg();
return 0;
}
案例 3:deque 实现滑动窗口(输出数组中每个窗口的最大值,简化版)
#include <deque>
#include <iostream>
#include <vector>
using namespace std;
void slidingWindowMax(const vector<int>& nums, int k) {
deque<int> dq;
for (int i = 0; i < nums.size(); i++) {
if (!dq.empty() && dq.front() < i - k + 1) {
dq.pop_front();
}
while (!dq.empty() && nums[dq.back()] < nums[i]) {
dq.pop_back();
}
dq.push_back(i);
if (i >= k - 1) {
cout << "窗口 [" << i - k + 1 << "," << i << "] 最大值:" << nums[dq.front()] << endl;
}
}
}
int main() {
vector<int> nums = {1,3,-1,-3,5,3,6,7};
int k = 3;
slidingWindowMax(nums, k);
return 0;
}
八、三者核心特性对比表格
为了方便大家快速梳理和记忆,整理了 stack、queue、deque 的核心特性对比:
| 特性 | stack(栈) | queue(队列) | deque(双端队列) |
|---|
| 数据结构类型 | 容器适配器 | 容器适配器 | 原生序列容器 |
| 访问规则 | 后进先出(LIFO),仅栈顶操作 | 先进先出(FIFO),队尾入、队头出 | 双端操作 + 随机访问 |
| 底层默认容器 | deque | deque | -(自身为底层容器) |
| 支持的核心操作位置 | 栈顶 | 队头(删、查)、队尾(插、查) | 头部、尾部、中间(随机访问) |
| 迭代器支持 | 不支持 | 不支持 | 支持随机访问迭代器 |
| 核心成员函数 | push()、pop()、top() | push()、pop()、front()、back() | push_front/back()、pop_front/back()、[]、at() |
| 清空方式 | 循环 pop() | 循环 pop() | clear()(直接支持) |
| 时间复杂度(核心操作) | O(1) | O(1) | 双端操作 O(1),随机访问 O(1) |
| 典型适用场景 | 括号匹配、函数调用栈 | 任务队列、BFS、排队模拟 | 滑动窗口、缓存队列、双端操作场景 |
总结
- deque 是原生双端队列,支持双端高效操作和随机访问,是 stack、queue 的默认底层容器,适合需要两端操作且偶尔随机访问的场景。
- stack 是后进先出适配器,仅支持栈顶操作,无迭代器,适合括号匹配、逆序输出等 LIFO 逻辑场景。
- queue 是先进先出适配器,支持队尾入、队头出,无迭代器,适合任务队列、BFS 等 FIFO 逻辑场景。
- 容器适配器(stack/queue)无 clear() 方法,需循环 pop() 清空,操作前务必用 empty() 判断是否为空,避免未定义行为。
对于 C++ 初学者而言,掌握这三个容器的核心用法,能灵活应对多数线性数据结构的应用场景,后续可结合更多实战案例,深化对 STL 容器的理解与运用。
微信扫一扫,关注极客日志
微信公众号「极客日志」,在微信中扫描左侧二维码关注。展示文案:极客日志 zeeklog
相关免费在线工具
- 加密/解密文本
使用加密算法(如AES、TripleDES、Rabbit或RC4)加密和解密文本明文。 在线工具,加密/解密文本在线工具,online
- Base64 字符串编码/解码
将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online
- Base64 文件转换器
将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online
- Markdown转HTML
将 Markdown(GFM)转为 HTML 片段,浏览器内 marked 解析;与 HTML转Markdown 互为补充。 在线工具,Markdown转HTML在线工具,online
- HTML转Markdown
将 HTML 片段转为 GitHub Flavored Markdown,支持标题、列表、链接、代码块与表格等;浏览器内处理,可链接预填。 在线工具,HTML转Markdown在线工具,online
- JSON 压缩
通过删除不必要的空白来缩小和压缩JSON。 在线工具,JSON 压缩在线工具,online