【C++】stack & queue

这篇具有很好参考价值的文章主要介绍了【C++】stack & queue。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、容器适配器

适配器是一种设计模式(设计模式是一套被反复使用的、多数人知晓的、经过分类编目的、代码设计经验的总结),该种模式是将一个类的接口转换成我们希望的另外一个接口。

虽然 stackqueue 中也可以存放元素,但在 STL 中并没有将其划分在容器的行列,而是将其称为容器适配器,这是因为 stackqueue 只是对其他容器的接口进行了包装,STLstackqueue 默认使用 deque(后面介绍), 比如:

【C++】stack & queue,C++,c++,开发语言,容器,stl

【C++】stack & queue,C++,c++,开发语言,容器,stl

其实容器适配器就是复用其他容器,利用其他容器的功能来适配出一个新的容器。

二、deque(了解)

deque(双端队列): 是一种双开口的 “连续” 空间的数据结构,双开口的含义是:可以在头尾两端进行插入和删除操作,且时间复杂度为 O(1),与 vector 比较,头插效率高,不需要搬移元素;与 list 比较,空间利用率比较高。如果需要高效的随机存取,还要大量的首尾的插入删除则建议使用 deque.

deque 并不是真正连续的空间,而是由一段段连续的小空间拼接而成的,实际 deque 类似于一个动态的二维数组,其底层结构如下图所示:

【C++】stack & queue,C++,c++,开发语言,容器,stl

vector 比较,deque 的优势是:头部插入和删除时,不需要搬移元素,效率特别高,而且在扩容时,也不需要搬移大量的元素,因此其效率是比 vector 高的;

list 比较,其底层是连续空间,空间利用率比较高,不需要存储额外字段;

但是,deque 有一个致命缺陷:不适合遍历,因为在遍历时,deque 的迭代器要频繁的去检测其是否移动到某段小空间的边界,导致效率低下,而序列式场景中,可能需要经常遍历,因此在实际中,需要线性结构时,大多数情况下优先考虑 vectorlistdeque 的应用并不多,而目前能看到的一个应用就是,STL 用其作为 stackqueue 的底层数据结构。

那么为什么选择 deque 作为 stackqueue 的底层默认容器呢?

stack 是一种后进先出的特殊线性数据结构,因此只要具有push_back()pop_back() 操作的线性结构,都可以作为 stack 的底层容器,比如 vectorlist 都可以;

queue先进先出的特殊线性数据结构,只要具有 push_backpop_front 操作的线性结构,都可以作为 queue 的底层容器,比如 list。但是 STL 中对 stackqueue 默认选择 deque 作为其底层容器,主要是因为:

  1. stackqueue 不需要遍历 (因此stack和queue没有迭代器),只需要在固定的一端或者两端进行操作。
  2. stack 中元素增长时,dequevector 的效率高(扩容时不需要搬移大量数据);queue 中的元素增长时,deque 不仅效率高,而且内存使用率高。结合了 deque 的优点,而完美的避开了其缺陷。

三、stack

1. stack 的介绍

我们先可以看一下 stack 的文档介绍:stack.

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

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

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

     		empty:判空操作
     		back:获取尾部元素操作
     		push_back:尾部插入元素操作
     		pop_back:尾部删除元素操作
    
  4. 标准容器 vector、deque、list 均符合这些需求,默认情况下,如果没有为 stack 指定特定的底层容器,默认情况下使用 deque.

我们先简单地看看 stack 的使用:

		void test_stack()
		{
			stack<int> st;
			st.push(1);
			st.push(2);
			st.push(3);
			st.push(4);
			st.push(5);
			while (!st.empty())
			{
				cout << st.top() << ' ';
				st.pop();
			}
			cout << endl;
		}

运行结果如下:

【C++】stack & queue,C++,c++,开发语言,容器,stl

2. 模拟实现 stack

我们使用 deque 作为 stack 的适配器模拟实现:

		#pragma once
		#include <vector>
		#include <deque>
		
		namespace Young
		{
			template <class T, class Container = deque<T>>
			class Stack
			{
			public:
				// 入栈
				void push(const T& val)
				{
					_con.push_back(val);
				}
		
				// 出栈
				void pop()
				{
					_con.pop_back();
				}
		
				// 获取栈顶元素
				T& top()
				{
					return _con.back();
				}
		
				// const 对象获取栈顶元素
				const T& top() const
				{
					return _con.back();
				}
				
				// 获取栈的大小
				size_t size()
				{
					return _con.size();
				}
		
				// 判断栈是否为空
				bool empty()
				{
					return _con.empty();
				}
		
			private:
				Container _con;
			};
		}

如上,stack 的常用接口就实现好了,我们再用我们自己实现的 stack 测试一下:

【C++】stack & queue,C++,c++,开发语言,容器,stl

四、queue

1. queue 的使用

我们先看一下 queue 的文档介绍:queue.

  1. 队列是一种容器适配器,专门用于在 FIFO 上下文(先进先出)中操作,其中从容器一端插入元素,另一端提取元素。

  2. 队列作为容器适配器实现,容器适配器即将特定容器类封装作为其底层容器类,queue 提供一组特定的成员函数来访问其元素。元素从队尾入队列,从队头出队列。

  3. 底层容器可以是标准容器类模板之一,也可以是其他专门设计的容器类。该底层容器应至少支持以下操作:

     		empty:检测队列是否为空
     		size: 返回队列中有效元素的个数
     		front:返回队头元素的引用
     		back: 返回队尾元素的引用
     		push_back:在队列尾部入队列
     		pop_front:在队列头部出队列
    
  4. 标准容器类 dequelist 满足了这些要求。默认情况下,如果没有为 queue 实例化指定容器类,则使用标准容器 deque.

先简单看一下 queue 的使用:

		void test_queue()
		{
			queue<int> q;
			q.push(1);
			q.push(2);
			q.push(3);
			q.push(4);
			q.push(5);
			while (!q.empty())
			{
				cout << q.front() << ' ';
				q.pop();
			}
			cout << endl;
		}

运行结果如下:

【C++】stack & queue,C++,c++,开发语言,容器,stl

2. 模拟实现 queue

我们也使用 deque 适配 queue

		#pragma once
		#include <deque>
		
		
		namespace Young
		{
			template<class T, class Container = deque<T>>
			class Queue
			{
			public:
				// 入队列
				void push(const T& val)
				{
					_con.push_back(val);
				}
		
				// 出队列
				void pop()
				{
					_con.pop_front();
				}
		
				// const 对象获取队头元素
				const T& front() const
				{
					return _con.front();
				}
		
				// 获取队头元素
				T& front()
				{
					return _con.front();
				}
		
				// const 对象获取队尾元素
				const T& back() const
				{
					return _con.back();
				}
		
				// 获取队尾元素
				T& back()
				{
					return _con.back();
				}
		
				// 获取队列长度
				size_t size()
				{
					return _con.size();
				}
		
				// 判断队列是否空
				bool empty()
				{
					return _con.empty();
				}
			private:
				Container _con;
			};
		}

我们再使用自己实现的 queue,来测试一下:

【C++】stack & queue,C++,c++,开发语言,容器,stl

3. priority_queue

(1)priority_queue 的介绍

priority_queue:优先级队列,是属于队列的一种,我们先看一下它的文档介绍 priority_queue.

  1. 优先级队列是一种容器适配器,根据严格的弱排序标准,它的第一个元素总是它所包含的元素中最大的。

  2. 此上下文类似于堆,在堆中可以随时插入元素,并且只能检索最大堆元素(优先级队列中位于顶部的元素)。

  3. 底层容器可以是任何标准容器类模板,也可以是其他特定设计的容器类。容器应该可以通过随机访问迭代器访问,并支持以下操作:

     		empty():检测容器是否为空
     		size(): 返回容器中有效元素个数
     		front():返回容器中第一个元素的引用
     		push_back():在容器尾部插入元素
     		pop_back(): 删除容器尾部元素
    
  4. 标准容器类 vectordeque 满足这些需求。默认情况下,如果没有为特定的 priority_queue 类实例化指定容器类,则使用vector

  5. 需要支持随机访问迭代器,以便始终在内部保持堆结构。容器适配器通过在需要时自动调用算法函数 make_heap、push_heap 和 pop_heap 来自动完成此操作。

(2)priority_queue 的使用

优先级队列默认使用 vector 作为其底层存储数据的容器,在 vector 上又使用了堆算法vector 中元素构造成堆的结构,因此 priority_queue 就是堆,所有需要用到堆的位置,都可以考虑使用priority_queue。注意:默认情况下 priority_queue大堆

【C++】stack & queue,C++,c++,开发语言,容器,stl

注意,我们看文档,默认情况下 priority_queue大堆,但是上图中红框中的是一个仿函数(后面介绍),就是实现比较的,其中 less 有小的意思,但是它却实现成大堆,这里要注意。其次,我们使用默认的参数时,只需要传第一个参数即可,后面的使用缺省参数即可,但是我们需要使用小堆的时候就需要将全部参数都传进去;我们先来看看使用:

		#include <vector>
		#include <queue>
		#include <functional> // greater算法的头文件
		void TestPriorityQueue()
		{
			// 默认情况下,创建的是大堆,其底层按照小于号比较
			vector<int> v{ 3,2,7,6,0,4,1,9,8,5 };
		
			priority_queue<int> q1;
			for (auto& e : v)
				q1.push(e);
		
			cout << q1.top() << endl;
		
		
			// 如果要创建小堆,将第三个模板参数换成 greater 比较方式
			priority_queue<int, vector<int>, greater<int>> q2(v.begin(), v.end());
			cout << q2.top() << endl;
		}

运行结果如下:

【C++】stack & queue,C++,c++,开发语言,容器,stl

(3)仿函数

仿函数又称为函数对象,是一个能行使函数功能的类。它的使用和我们平时的函数调用一样。

首先我们得先实现一个类,这个类中需要实现 () 运算符重载,里面实现的功能需要我们自己实现,假设我们需要实现 priority_queue 中的大堆,如下:

		// 仿函数 --- 大堆,大的优先级大
		template <class T>
		class Less
		{
		public:
			bool operator()(const T& x, const T& y)
			{
				return x < y;
			}
		};

那么如何调用呢?首先我们得先创建一个对象,再使用这个对象进行调用函数:

			Less<int> less;
			cout << less(1, 8) << endl; // 与下等价
			cout << less.operator()(1, 8) << endl;

运行结果如下:

【C++】stack & queue,C++,c++,开发语言,容器,stl

下面我们使用仿函数的形式模拟实现 priority_queue.

(4)模拟实现 priority_queue

	#pragma once
	#include <vector>
	
	namespace Young
	{
		// 模板参数
		template <class T, class Container = vector<T>, class Compare = Less<T>>
		class PriorityQueue
		{
		public:
			// 向上调整 --- 大堆
			void adjust_up(size_t child)
			{
				Compare com;
				size_t parent = (child - 1) / 2;
				while (child > 0)
				{
					//if (_con[parent] < _con[child]) // 与下等价
					if (com(_con[parent], _con[child]))
					{
						swap(_con[child], _con[parent]);
						child = parent;
						parent = (child - 1) / 2;
					}
					else
					{
						break;
					}
				}
			}
	
			// 向下调整 --- 大堆
			void adjust_down(size_t parent)
			{
				Compare com;
				size_t child = parent * 2 + 1;
				
				while (child < _con.size())
				{
					if (child + 1 < _con.size() && _con[child + 1] > _con[child])
					{
						++child;
					}
	
					// if (_con[parent] < _con[child])  // 与下等价
					if (com(_con[parent], _con[child]))
					{
						swap(_con[child], _con[parent]);
						parent = child;
						child = parent * 2 + 1;
					}
					else
					{
						break;
					}
				}
			}
	
			// 入数据
			void push(const T& val)
			{
				_con.push_back(val);
				adjust_up(_con.size() - 1);
			}
	
			// 出数据
			void pop()
			{
				swap(_con[0], _con[_con.size() - 1]);
				_con.pop_back();
	
				adjust_down(0);
			}
	
			// 获取优先级最大的数据
			const T& top()
			{
				return _con.front();
			}
	
			// 判空
			bool empty()
			{
				return _con.empty();
			}
	
		private:
			Container _con;
		};
	}

下面我们使用自己实现的优先级队列进行测试:

		void test_priority_queue()
		{
			// Young::PriorityQueue<int, vector<int>, Greater<int>> pq; // 小堆
		
			// Young::PriorityQueue<int, vector<int>, Less<int>> pq; // 大堆,与下等价
			Young::PriorityQueue<int> pq;  // 缺省参数默认是大堆
			pq.push(2);
			pq.push(8);
			pq.push(1);
			pq.push(0);
			pq.push(10);
		
			while (!pq.empty())
			{
				cout << pq.top() << ' ';
				pq.pop();
			}
		
			cout << endl;
		}

测试结果如下:

【C++】stack & queue,C++,c++,开发语言,容器,stl文章来源地址https://www.toymoban.com/news/detail-726677.html

到了这里,关于【C++】stack & queue的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 容器适配器---deque和STL ---stack queue priority_queue的模拟实现 C++

    目录 一、容器适配器 deque原理 deque的缺陷 deque的优势 二、stack的模拟实现  三、queue的模拟实现 四、优先级队列的模拟实现 适配器是一种设计模式(设计模式是一套被反复使用的、多数人知晓的、经过分类编目的、代码设计经验的总结),该种模式是将一个类的接口转换成客户

    2024年02月02日
    浏览(10)
  • 【C++】STL中的容器适配器 stack queue 和 priority_queue 的模拟实现

    【C++】STL中的容器适配器 stack queue 和 priority_queue 的模拟实现

    适配器是一种设计模式 (设计模式是一套被反复使用的、多数人知晓的、经过分类编目的、代码设计经验的总结),该种模式是将一个类的接口转换成客户希望的另外一个接口。 例如我们常见的充电器就是一种适配器,它将我们常用的220V交流电压转化为4,5V (或者其他更高的电

    2023年04月26日
    浏览(12)
  • STL——stack容器、queue容器、list容器

    STL——stack容器、queue容器、list容器

    就是栈 栈不允许有遍历行为 是先进后出的数据结构 是先进先出的数据结构 **队列容器允许从一端新增元素,从另一端移除元素 队列中只有队头和队尾才可以被外界使用,因此队列不允许有遍历行为 队列中进数据称为—入队push 队列中出数据称为—出队pop ** 在STL中这个链表

    2024年02月09日
    浏览(9)
  • STL——stack容器和queue容器详解

    STL——stack容器和queue容器详解

      目录 💡stack 💡基本概念 常用接口  💡queue 💡基本概念 💡常用接口 栈(stack): 一种特殊的线性表,其只允许在固定的一端进行插入和删除操作。在进行数据插入和删除的一端称为栈顶,另一端称为栈底。栈中的元素都遵循后进先出的原则(LIFO,Last In First Out)。 压栈

    2024年02月02日
    浏览(9)
  • STL: 容器适配器stack 与 queue

    STL: 容器适配器stack 与 queue

      目录 1.容器适配器 1.1 STL标准库中stack和queue的底层结构 1.2 deque的简单介绍(了解) 1.2.1 deque的原理介绍 1.2.2 deque的缺陷 1.2.3 为什么选择deque作为stack和queue的底层默认容器 2. stack的介绍和使用 2.1 stack的介绍  2.2 stack的使用 2.3 利用deque模拟实现stack 3.queue的介绍和使用 3.1 queue的

    2024年02月05日
    浏览(13)
  • C++ STL stack & queue

    C++ STL stack & queue

    目录 一.stack 介绍  二.stack 使用 三.stack 模拟实现 普通版本: 适配器版本: 四.queue的介绍 五. queue使用 六.queue模拟实现 七.deque介绍 1.容器适配器 2.deque的简单介绍 3.deque的缺陷 4.为什么选择deque作为stack和queue的底层默认容器 stack------reference 1. stack是一种容器适配器,专门用在

    2024年02月12日
    浏览(9)
  • 【STL】容器适配器stack和queue常见用法及模拟实现

    【STL】容器适配器stack和queue常见用法及模拟实现

    1.stack介绍及使用 1.1 stack的介绍 stack文档介绍 stack是一种容器适配器,专门用在具有后进先出操作的上下文环境中,其删除只能从容器的一端进行元素的插入与提取操作。 stack是作为容器适配器被实现的,容器适配器是使用特定容器类的封装对象作为其基础容器的类,提供一

    2024年02月06日
    浏览(12)
  • C++ STL--->stack和queue

    C++ STL--->stack和queue

    stack文档 stack是一种容器适配器,专门用在具有后进先出操作的上下文环境中,其删除只能从容器的一端进行 元素的插入与提取操作。 stack是作为容器适配器被实现的,容器适配器即是对特定类封装作为其底层的容器,并提供一组特定 的成员函数来访问其元素,将特定类作为

    2024年01月16日
    浏览(22)
  • [ C++ ] STL---stack与queue

    [ C++ ] STL---stack与queue

    目录 stack简介 stack的常用接口 queue简介 queue的常用接口 stack的模拟实现 queue的模拟实现 1. stack是具有后进先出操作的一种容器适配器 ,其只能从容器的一端进行元素的插入与删除操作 ; 2. stack是作为容器适配器被实现的,容器适配器即是对特定类封装作为其底层的容器,并

    2024年03月28日
    浏览(10)
  • 【C++】学习STL中的stack和queue

            今天这篇博客的内容主要关于STL中的stack、queue和priority_queue三种容器。         stack和queue的使用方式非常简单,我们只要根据之前学习数据结构的经验和文档介绍就可以轻松上手。于是我们直接开始对它们的模拟实现。         stack和queue我们在数据结构阶段就曾经

    2024年02月10日
    浏览(7)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包