【C++】-- STL容器适配器之stack

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介: 【C++】-- STL容器适配器之stack

一、适配器

适配器是一种设计模式,能够将一个类的接口转换成客户希望的另外一个接口,从而使原本接口不匹配而无法在一起工作的两个类能够在一起工作。比如对于笔记本来说,电源额定电压是220V,而美国电压是110V,为了能在美国使用,必须要用变压器转换电压以匹配美国电压,那么这个变压器就是个适配器。

容器适配器是一个封装了序列容器的类模板,它在一般序列容器的基础上提供了一些不同的功能。它可以通过适配容器现有的接口来提供不同的功能,所以叫作适配器类。

虽然stack和queue中也可以存放元素,但在STL中并没有将其划分在容器的行列,而是将其称为容器适配器,这是因为stack和队列只是对其他容器的接口进行了包装,STL中stack和queue默认使用deque。

二、栈

1.栈的性质

(1)stack是一种容器适配器,专门用在具有后进先出操作的上下文环境中,其删除只能从容器的一端进行元素的插入与提取操作。

(2) stack是作为容器适配器被实现的,容器适配器即是对特定类封装将其作为底层的容器,并提供一组特定的成员函数来访问其元素,将特定类作为其底层,元素从特定容器的尾部(即栈顶)被压入和弹出。

(3)stack的底层容器可以是任何标准的容器类模板或者一些其他特定的容器类,这些容器类应该支持以下操作:

       empty:判空操作

       back:获取尾部元素操作

       push_back:尾部插入元素操作

       pop_back:尾部删除元素操作

(4) 标准容器vector、deque、list均符合这些需求,默认情况下,如果没有为stack指定特定的底层容器, 默认情况下使用deque。

2.栈类

(1)栈的构造

构造一个栈

explicit stack (const container_type& ctnr = container_type());//构造一个栈的容器适配器对象

构造一个栈st1:

stack<int> st1;//构造一个空栈st1

(2)empty( )

判断栈是否为空

bool empty() const;//判断栈是否为空

判断st1是否为空:

cout << st1.empty() << endl;

(3)push( )

从栈顶插入元素

void push (const value_type& val);//向栈顶插入元素值为val的元素

向栈顶插入值分别为1,2,3,4的4个元素

1.  st1.push(1);
2.  st1.push(2);
3.  st1.push(3);
4.  st1.push(4);

F10监视可以看到 st1有4个元素:

(4)pop( )

从栈顶删除元素

void pop();
st1.pop();//删除st1栈顶元素4

执行pop后,st1只有3个元素,4被删除了

(5)top( )

返回栈顶元素引用

1. value_type& top();
2. const value_type& top() const;

修改栈顶元素,现在栈顶元素是3:

st1.top() += 20;

执行后变成了23:

(6)size( )

求栈的元素个数:

size_type size() const;

求st1栈的元素个数:

cout << st1.size() << endl;

三、栈的模拟实现

栈作为容器适配器, 不像string、vector、list等是完整的容器类,栈不是完整的容器类,而是提供特定接口的类,相当于在完整容器外面包装了一层,比如使用vector实现栈,就可以借助vector的操作来实现栈的操作。

栈的构造函数、拷贝构造函数、赋值运算符载函数、析构函数不需要写,会调用vector自己的默认构造函数、拷贝构造函数、赋值运算符重载函数、析构函数。

016-stack.h

1. #pragma once
2. #include<iostream>
3. #include<vector>
4. 
5. namespace delia
6. {
7.  //容器适配器
8.  template<class T,class Container  = std::vector<T>>//这里指定底层容器为vector,而底层也可以不用vector实现,也可以用list实现
9.  class stack
10.   {
11.   public:
12.     //栈的插入
13.     void push(const T& x)
14.     {
15.       _con.push_back(x);
16.     }
17. 
18.     //栈的删除
19.     void pop()
20.     {
21.       _con.pop_back();
22.     }
23. 
24.     //返回栈顶元素
25.     T top()
26.     {
27.       return _con.back();
28.     }
29. 
30.     //求栈中元素的个数
31.     size_t size()
32.     {
33.       return _con.size();
34.     }
35. 
36.     //判断栈是否为空
37.     bool empty()
38.     {
39.       return _con.empty();
40.     }
41. 
42.   private:
43.     Container _con;
44.   };
45. 
46. }

016-test.cpp

1. #include<iostream>
2. 
3. using std::cout;
4. using std::endl;
5. 
6. #include "016-stack.h"
7. 
8. void test_stack()
9. {
10.   delia::stack<int> st;
11.   st.push(1);
12.   st.push(2);
13.   st.push(3);
14.   st.push(4);
15. 
16.   while (!st.empty())
17.   {
18.     cout << st.top() << " ";//打印栈顶元素
19.     st.pop();//弹出栈顶元素
20.   }
21.   cout << endl;
22. }
23. 
24. int main()
25. {
26.   test_stack();
27.   return 0;
28. }

打印结果:

假如底层不想用vector实现,可以用list实现,只需要修改第3行和第8行:

1. #pragma once
2. #include<iostream>
3. #include<list>
4. 
5. namespace delia
6. {
7.  //容器适配器
8.  template<class T,class Container  = std::list<T>>//这里指定底层容器为vector,而底层也可以不用vector实现,也可以用list实现
9.  class stack
10.   {
11.   public:
12.     //栈的插入
13.     void push(const T& x)
14.     {
15.       _con.push_back(x);
16.     }
17. 
18.     //栈的删除
19.     void pop()
20.     {
21.       _con.pop_back();
22.     }
23. 
24.     //返回栈顶元素
25.     T top()
26.     {
27.       return _con.back();
28.     }
29. 
30.     //求栈中元素的个数
31.     size_t size()
32.     {
33.       return _con.size();
34.     }
35. 
36.     //判断栈是否为空
37.     bool empty()
38.     {
39.       return _con.empty();
40.     }
41. 
42.   private:
43.     Container _con;
44.   };
45. 
46. }


相关文章
|
5天前
|
存储 C++ 容器
C++:STL - set & map
C++:STL - set & map
14 4
|
6天前
|
设计模式 算法 编译器
【C++】开始使用stack 与 queue
队列的相关习题大部分是子啊BFS中使用,这里就不在说明了
12 3
|
6天前
|
调度 C++ 容器
【C++】手搓 list 容器
本文我们实现了STL库中重要的list 的模拟实现,其中最重要莫过于迭代器的封装类的书写,这是前所未有的操作(对于我来说,我是第一次使用这种结构)。通过list 的模拟实现也帮我们巩固了类与对象的知识,也强化了指针操作的思路。欢迎大家讨论分析。
12 1
|
6天前
|
算法 安全 程序员
【C++】STL学习之旅——初识STL,认识string类
现在我正式开始学习STL,这让我期待好久了,一想到不用手撕链表,手搓堆栈,心里非常爽
15 0
|
7天前
|
存储 Serverless C++
【C++入门到精通】哈希 (STL) _ unordered_map _ unordered_set [ C++入门 ]
【C++入门到精通】哈希 (STL) _ unordered_map _ unordered_set [ C++入门 ]
10 1
|
8天前
|
存储 设计模式 算法
【C++/STL】stack和queue(容器适配器、优先队列、双端队列)
【C++/STL】stack和queue(容器适配器、优先队列、双端队列)
13 1
|
6天前
|
监控 Kubernetes Docker
【Docker 专栏】Docker 容器内应用的健康检查与自动恢复
【5月更文挑战第9天】本文探讨了Docker容器中应用的健康检查与自动恢复,强调其对应用稳定性和系统性能的重要性。健康检查包括进程、端口和应用特定检查,而自动恢复则涉及重启容器和重新部署。Docker原生及第三方工具(如Kubernetes)提供了相关功能。配置检查需考虑检查频率、应用特性和监控告警。案例分析展示了实际操作,未来发展趋势将趋向更智能和高效的检查恢复机制。
【Docker 专栏】Docker 容器内应用的健康检查与自动恢复
|
2天前
|
Prometheus 监控 Cloud Native
构建高效稳定的Docker容器监控体系
【5月更文挑战第13天】在微服务架构和容器化部署日益普及的背景下,对Docker容器的监控变得尤为重要。本文将探讨一种构建高效稳定Docker容器监控体系的方法,通过集成Prometheus和cAdvisor工具,实现对容器资源使用情况、性能指标和运行状态的实时监控。同时,结合Grafana进行数据可视化,为运维人员提供直观的分析界面,以便及时发现和解决潜在问题,保障系统的高可用性和稳定性。
15 6
|
2天前
|
存储 安全 开发者
如何删除 Docker 镜像、容器和卷?
【5月更文挑战第11天】
13 2
如何删除 Docker 镜像、容器和卷?
|
4天前
|
NoSQL Redis Docker
Mac上轻松几步搞定Docker与Redis安装:从下载安装到容器运行实测全程指南
Mac上轻松几步搞定Docker与Redis安装:从下载安装到容器运行实测全程指南
19 0