STL stack,queue,deque以及适配器

目录

stack

stack的使用

下面是stack库中的接口函数,有了前面的基础,我们可以根据函数名得知函数的作用

函数 说明
stack() 构造空栈
empty() 判断栈是否为空
size() 返回栈中元素个数
top 返回栈顶元素
push() 将值从栈顶压入栈内
pop() 在栈顶出栈

stack模拟实现

栈其实就是一种特殊的vector,因此可以使用vector模拟实现stack

比较容易:

cpp 复制代码
#include <iostream>
#include <vector>
using namespace std;

namespace my_stack
{
	template<class T>
	class stack
	{
	public:
		void push(const T& val)
		{
			_v.push_back(val);
		}

		void pop()
		{
			_v.pop_back();
		}

		T& top()
		{
			return _v.back();
		}

		bool empty()
		{
			return _v.empty();
		}

		size_t size()
		{
			return _v.size();
		}


	private:
		vector<T> _v;
	};
}

queue

queue的使用

函数 说明
queue() 构造空队列
empty() 判断队列是否为空
size() 返回队列中元素个数
front() 返回队头元素的引用
back() 返回队尾元素的引用
push() 在队尾入队
pop() 在队头出队

queue模拟实现

queue的接口中存在头删,用vector实现起来效率太低,所以可以使用list实现

cpp 复制代码
#include <iostream>
#include <list>
using namespace std;


namespace my_queue
{
	template<class T>
	class queue
	{
	public:
		void push(const T& val)
		{
			_lt.push_back(val);
		}

		void pop()
		{
			_lt.pop_front();
		}

		T& front()
		{
			return _lt.front();
		}

		T& back()
		{
			return _lt.back();
		}

		bool empty()
		{
			return _lt.empty();
		}

		size_t size()
		{
			return _lt.size();
		}

	private:
		list<T> _lt;
	};
}

适配器

适配器是一种设计模式,该模式是将一个类的接口转换成客户希望的另一个接口

对已有的东西,进行适配转换

在C语言中,我们习惯用顺序表去实现栈,因为使用顺序表实现栈比较方便

但是也可以用链表去实现,我们如果想去用链表实现栈,还需要再写一套,会比较麻烦

在C++中,有了适配器,不用再实现两遍了

我们可以使用适配器进行实现,这样我们就可以做到数组栈和链表栈秒切换了

在用适配器实现后,平时使用时感觉不到差异,但是两者的底层逻辑完全不同

下面我们用适配器设计出stack

首先我们添加一个模板参数
template<class T,class Container>Container就是适配器

在实例化的时候,我们需要指定适配器是哪一种容器

接下来,把成员变量改为适配器

cpp 复制代码
	template<class T,class Container>
	class stack
	{
	public:
	
	private:
		Container _con;
	};

接下来,我们按照前面模拟实现的stack把所有_v改为_con就可以了

cpp 复制代码
namespace my_stack
{
	template<class T,class Container>
	class stack
	{
	public:
		void push(const T& val)
		{
			_con.push_back(val);
		}

		void pop()
		{
			_con.pop_back();
		}

		T& top()
		{
			return _con.back();
		}

		bool empty()
		{
			return _con.empty();
		}

		size_t size()
		{
			return _con.size();
		}


	private:
		Container _con;
	};
}

接下来我们看一下 用适配器实现的数组栈和链表栈

cpp 复制代码
int main()
{
	my_stack::stack<int, vector<int>> st;
	st.push(1);
	st.push(2);
	st.push(3);
	st.push(4);
	st.push(5);
	st.push(6);

	while (!st.empty())
	{
		cout << st.top() << " ";
		st.pop();
	}//输出 6 5 4 3 2 1
	cout << endl;


	my_stack::stack<int, list<int>> st1;
	st1.push(1);
	st1.push(2);
	st1.push(3);
	st1.push(4);
	st1.push(5);
	st1.push(6);

	while (!st1.empty())
	{
		cout << st1.top() << " ";
		st1.pop();
	}
	//输出 6 5 4 3 2 1
	cout << endl;
}

下面我们也可以使用适配器实现queue

cpp 复制代码
namespace my_queue
{
	template<class T, class Container>
	class queue
	{
	public:
		void push(const T& val)
		{
			_con.push_back(val);
		}

		void pop()
		{
			_con.pop_front();
		}

		T& front()
		{
			return _con.front();
		}

		T& back()
		{
			return _con.back();
		}

		bool empty()
		{
			return _con.empty();
		}

		size_t size()
		{
			return _con.size();
		}

	private:
		Container _con;

	};
}

这里的适配器类型只能是list,不能是vector因为vector中不支持pop_front头删,无法支持queuepop

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


deque

我们看STL中,stackqueue的实现

可以发现它们的适配器默认为deque<T>

这个deque是什么容器,我们下面来看一看

deque叫双端队列(但不是队列),是一种双开口的"连续空间"的数据结构

deque可以在头尾两端进行插入和删除操作,时间复杂度为O(1),效率高,并且支持[]

deque并不是真正连续的空间,而是由一段段连续的小空间拼接而成的,实际deque类似于一个动态的二维

数组

deque中有一段中控数组(本质是指针数组),其中每一个元素指针都指向一个固定大小的缓冲区

插入元素先从存入中控数组中间位置指针指向的缓冲区,因为要保证头插和尾插都有空间

如果一个Buff满了,就在其下一个指针指向的Buff中存储数据

如果中控数组满了,扩容即可,对于指针类型的拷贝消耗少

deque相比于vector

极大缓解了扩容的问题,同时解决了头删和头插带来的效率低的问题

但是deque的[]随机访问相对于vector的随机访问还是有差距的,因为要计算随机访问的位置在哪个Buff的哪个位置上

假设要访问位置i上的元素

1.先看i在不在第一个Buff中,如果在就直接找到位置访问

2.如果不在第一个Buff,i-=第一个Buffsize()

在第i/buffsizeBuff

在这个Buffi%buffsize位置上

而vector中的[]就是直接解引用指针,效率高

deque相比于list:
deque支持随机访问

cpu高速访问效率搞

但是deque中间位置元素的插入和删除效率没有list

所以deque的最大价值就在于它的头插,头删,尾插,尾删的效率高

这也正是stackqueue中适配器最需要的特点,所以deque作为stackqueue的默认适配器最合适

用deque为默认适配器实现stackqueue

cpp 复制代码
#include <deque>
#include <stack>
#include <list>
#include <iostream>
using namespace std;
namespace my_queue
{
	template<class T, class Container = deque<T>>
	class queue
	{
	public:
		void push(const T& val)
		{
			_con.push_back(val);
		}

		void pop()
		{
			_con.pop_front();
		}

		T& front()
		{
			return _con.front();
		}

		T& back()
		{
			return _con.back();
		}

		bool empty()
		{
			return _con.empty();
		}

		size_t size()
		{
			return _con.size();
		}

	private:
		Container _con;

	};
}


namespace my_stack
{
	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();
		}

		bool empty()
		{
			return _con.empty();
		}

		size_t size()
		{
			return _con.size();
		}


	private:
		Container _con;
	};
}
相关推荐
软件黑马王子1 小时前
C#初级教程(4)——流程控制:从基础到实践
开发语言·c#
闲猫2 小时前
go orm GORM
开发语言·后端·golang
黑不溜秋的3 小时前
C++ 设计模式 - 策略模式
c++·设计模式·策略模式
李白同学3 小时前
【C语言】结构体内存对齐问题
c语言·开发语言
黑子哥呢?4 小时前
安装Bash completion解决tab不能补全问题
开发语言·bash
青龙小码农4 小时前
yum报错:bash: /usr/bin/yum: /usr/bin/python: 坏的解释器:没有那个文件或目录
开发语言·python·bash·liunx
大数据追光猿4 小时前
Python应用算法之贪心算法理解和实践
大数据·开发语言·人工智能·python·深度学习·算法·贪心算法
Dream it possible!5 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
夏末秋也凉5 小时前
力扣-回溯-46 全排列
数据结构·算法·leetcode
柠石榴5 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯