【C++自学笔记】STL—详细理解stcak和queue、priority_queue 及 其模拟实现(容器适配器)

一、stack的介绍和使用

1、stack的介绍

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

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

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

  • empty:判空操作
  • back:获取尾部元素操作
  • push_back:尾部插入元素
  • pop_back:尾部删除元素

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

2、stack的使用

函数说明接口说明

stack(const container_typr& ctnr = container_typr())

构造空的栈

bool empty() const

检测stack是否为空

size_type size() const返回stack中元素的个数
value_type& top()返回栈顶的元素
const value_type& top() const

返回栈顶元素的const引用

void push(const value_type& val)将元素val压栈
void pop()

将stack中尾部元素弹出

tempalce<class... Args>

void empale(Args&&... args)

将stack的栈顶构造元素
void swap(stack& x)交换两个栈中的元素

 

二、queue的介绍和使用

1、queue的介绍

1、队列是一种容器配置器,专门用于在FIFO(先入先出)中操作,其中从容器的一段进入元素,另一端提取元素;

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

3、底层容器可以是标准容器类模板之一,该底层容器至少支持以下操作:

  • empty:判断队列是否为空
  • size:返回队列中有效元素的个数
  • front:返回队头元素的引用
  • back:返回队尾元素的引用
  • push_back:在队列尾部入队列
  • pop_front:在队列头部出队列

4、标准容器 deque 和 list 都满足了要求,但是默认情况下使用 deque;

2、queue的使用

函数声明

接口说明

queue(const container_type& ctnr = container_type())构造空的队列
bool empty()检测队列是否为空,
size_type size() cosnt返回队列中的有效元素个数
value_type& front()返回队头元素的引用
const value_type& front() const返回队头元素的const 引用
value_type& back()返回队尾元素的引用
const value_type& back() const返回队尾元素的const 引用
void push(value_type& val)在队尾插入 val元素
void pop()

将队头元素出列

template<class... Args>

void emplace(Args&&... args)

在队尾构造元素
void swap(queue& x)交换两个队列中的元素

三、priority_queue是介绍和使用

1、priority_queue的介绍

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

2、类似于堆,在堆中可以随时插入元素,并且只能检索最大堆元素(优先队列中位于顶部的元素);

3、优先队列被实现为容器适配器;

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

  • empty:判空操作
  • back:获取尾部元素操作
  • push_back:尾部插入元素
  • pop_back:尾部删除元素

5、标准容器类 vector 和 deque 都可以用来是先priority_queue,默认情况下使用vector;

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

2、priority_queue的使用

优先级队列默认使用 vector 作为其底层存储数据的容器,在vector 上有使用了堆算法将 vector 中元素构造成堆的结构,因此 priority_queue就是被认为时堆,所有需要用到堆的地方都可以使用priority_queue来替代。(默认是大堆结构)

函数声明接口说明
priority_queue(const Compare& x= Compare(),const Container& y = Container())构造一个空的优先级队列

template

priority_queue(Inputlterator first,Inputlterator last,const Compare& comp = Compare(),const Container& ctnr = Container())

用[fist,last)区间构造优先级队列
bool empty() const判空
const value_type& top() const返回优先级中最大的元素
void push(const T& x)插入x
void pop()删除对顶元素

 

3、pirority_queue的是使用注意点:

1、默认情况下,priority_queue是大堆;

void Test1() {
	//默认使用大堆结构
	vector<int> v{3, 2, 7, 6, 0, 4, 1, 9, 8, 5};
	priority_queue<int> q1;
	for (auto x : v)
		q1.push(x);
	cout << q1.top() << endl;

	//如果需要小堆,将第三个参数模板换成 greater 的比较方式
	priority_queue<int, vector<int>, greater<int>> q2(v.begin(),v.end());
	cout << q2.top << endl;
}

2、如果在 priority_queue 中放自定义类型的数据,用户需要在自定义类型中提供 < or  > 的重载;


class Date {
public:
	Date(int year = 1900, int mouth = 1, int day = 1)
		:_year(year)
		, _mouth(mouth)
		, _day(day)
	{}
	bool operator<(const Date& d) const {
		return (_year < d._year) ||
			(_year == d._year && _mouth < d._mouth) ||
			(_year == d._year && _mouth == d._mouth && _day < d._day);
	}
	bool operator>(const Date& d)const {
		return (_year > d._year) ||
			(_year == d._year && _mouth > d._mouth) ||
			(_year == d._year && _mouth == d._mouth && _day > d._day);
	}
	friend ostream& operator<<(ostream&& _cout, const Date& d) {
		_cout << d._year << "-" << d._mouth << "-" << d._day;
		return _cout;
	}
private:
	int _year;
	int _mouth;
	int _day;
};
void Test2(){
	//大堆
	priority_queue<Date> q1;
	q1.push(Date(2018, 1, 1));
	q1.push(Date(2018, 2, 2));
	q1.push(Date(2018, 3, 3));



	//小堆
	priority_queue<Date, vector<Date>, greater<Date>> q2;
	q2.push(Date(2017, 1, 1));
	q2.push(Date(2017, 2, 2));
	q2.push(Date(2017, 3, 3));

}

3、有些情况下,用户可能需要提供比较器规则;

class Less {
public:
	bool operator()(const Date* pLeft,const Date* pRight) {
		return *pLeft < *pRight;
	}
};
void Test3() {
	priority_queue<Date*, vector<Date*>, Less> q;
	q.push(&Date(2018, 1, 1));
	q.push(&Date(2018, 2, 2));
	q.push(&Date(2018, 3, 3));

}

四、模拟实现 stack 和 queue

1、stack的模拟实现(deque)

namespace My {
	template<class T,class Con = deque<T>>
	class Stack {
	public:
		Stack(){}
		void push(const T& x) {
			_c.push_back(x);
		}
		void pop() {
			_c.pop_back();
		}
		T& Top{
			return _c.back();
		}
		const T& Top() const {
			return _c.back();
		}
		size_t Size() const {
			return _c.size();
		}
		bool Empty() const {
			return _c.empty();
		}
	private:
		Con _c;
	};
}

2、queue的模拟实现(deque)

 

namespace My {
	template<class T,class Con = deque<T>>
	class Queue {
	public:
		Queue(){}
		bool Empty() const {
			return _c.empty();
		}
		size_t Size() const {
			return _c.size();
		}
		T& Front() {
			return _c.front();
		}
		const T& Front() cosnt {
			return _c.front();
		}
		T& back() {
			return _c.back();
		}
		const T& back() cosnt {
			return _c.back();
		}
		void Pop() {
			_c.pop_front();
		}
		void Push(const T& x) {
			_c.push_back(x);
		}
	private:
		Con _c;
	};
}

3、为什么选择deque作为 stack 和 queue 的底层默认容器(常见面试题)

stack 是一种 LIFO 的特殊线性结构,因此只要具有 push_back 和 pop-back 操作的线性结构,都可以作为 stack 的底层容器;queue 是一种 FIFO 的特殊线性结构,只要具有 push_back 和 pop_front 操作的线性结构都可以;

但是STL中对 stack 和 queue 的底层默认选用 deque 的主要原因是:

  1. stack 和 queue 不需要遍历(因此 stack 和 queue 没有迭代器),只需要在固定的一端或两端进行操作即可;
  2. 在 stack 和 queue 中元素增长时,deque 比 vector 的效率高;queue 中的元素增长时,deque 不仅效率高,而且内存使用率高;
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
C++ STL中的priority_queue是一个优先队列,它是一个使用堆来实现容器。它可以按照一定的优先级顺序存储元素,并且每次访问队首元素都是访问优先级最高的元素。 在使用priority_queue时,可以通过定义不同的比较函数来指定元素的优先级顺序。默认情况下,对于基本类型,默认是大顶堆,降序队列。也可以通过指定参数来实现小顶堆,升序队列。例如: priority_queue<int, vector<int>, greater<int>> q; //小顶堆,升序队列 priority_queue<int, vector<int>, less<int>> q; //大顶堆,降序队列 在对priority_queue进行操作时,可以使用push()函数向队列中插入元素,使用top()函数获取队首元素,使用pop()函数删除队首元素。 在自定义类型的优先队列中,可以重载运算符>或<来定义优先级。例如,可以重载operator>来定义小顶堆,即优先级较小的元素排在前面。示例代码如下: struct Node{ int x, y; Node(int a=0, int b=0): x(a), y(b) {} }; bool operator> (Node a, Node b){ if(a.x == b.x) return a.y > b.y; return a.x > b.x; } priority_queue<Node, vector<Node>, greater<Node>> q; q.push(Node(rand(), rand())); while(!q.empty()){ cout<<q.top().x<<' '<<q.top().y<<endl; q.pop(); } 总结来说,C++ STL中的priority_queue是一个使用堆实现的优先队列,可以按照指定的优先级顺序存储元素。可以通过定义不同的比较函数或重载运算符来指定优先级规则。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [【总结】C++ 基础数据结构 —— STL之优先队列(priority_queue) 用法详解](https://blog.csdn.net/weixin_44668898/article/details/102132580)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

_ClivenZ

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值