- 【写在前面】
- 一、stack的介绍及使用
- 💦 stack的介绍
- 💦 stack的使用
- 💦 stack的OJ
- 1、最小栈<难度系数⭐>
- 2、栈的d出压入序列<难度系数⭐⭐>
- 3、逆波兰表达式求值<难度系数⭐⭐>
- 4、用栈实现队列<难度系数⭐>
- 4、用队列实现栈<难度系数⭐>
- 💦 stack的模拟实现
- 二、queue的介绍及使用
- 💦 queue的介绍
- 💦 queue的使用
- 💦 queue的模拟实现
- 三、priority_queue的介绍及使用
- 💦 priority_queue的介绍
- 💦 priority_queue的使用
- 💦 priority_queue的OJ
- 1、数组中的第K个最大元素<难度系数⭐⭐>
- 💦 priority_queue的模拟实现
- 💦 仿函数的变异玩法
- 四、容器适配器
- 💦 什么是适配器
- 💦 STL标准库中stack和queue的底层结构
- 💦 deque的简单介绍(了解)
- 1、deque的简单使用
- 2、deque的原理介绍
- 2、deque的缺陷及优点
- 💦 为什么选择deque作为stack和queue的底层默认容器
- 💦 STL标准库中对于stack和queue的模拟实现
- 1、stack的模拟实现
- 2、queue的模拟实现
虽然 cplusplus 把 stack 和 queue 归类到了 Containers 下,但是严格来说 stack and queue 不再是容器了,而属于容器适配器 or 容器配接器,适配器做的功能是转换 —— 它不是直接实现的,而是由其它容器封装转换实现的,在下面的模拟实现我们会细谈。
它做为容器适配器,它与容器有一个具大的差别之一就是它没有迭代器,不是说它不能实现迭代器,而是没有必要实现迭代器,因为它如果实现了迭代器,就没法保障 stack “Last In First Out” 和 queue “First In First Out” 的原则。
其次对于 stack 和 queue 的使用比较简单,我们大概过一下,以 OJ 的形式来了解它们。
stack文档介绍
- stack 是一种容器适配器,专门用在具有后进先出 *** 作的上下文环境中,其删除只能从容器的一端进行元素的插入与提取 *** 作。
- stack 是作为容器适配器被实现的,容器适配器即是对特定类封装作为其底层的容器,并提供一组特定的成员函数来访问其元素,将特定类作为其底层,元素特定容器的尾部(即栈顶)被压入和d出。
- stack 的底层容器可以是任何标准的容器类模板或者一些其他特定的容器类,这些容器类应该支持以下 *** 作:
➡ empty:判空 *** 作
➡ back:获取尾部元素 *** 作
➡ push_back:尾部插入元素 *** 作
➡ pop_back:尾部删除元素 *** 作 - 标准容器 vector、deque、list 均符合这些需求,默认情况下,如果没有为 stack 指定特定的底层容器,默认情况下使用 deque。
函数声明 | 接口说明 |
---|---|
stack() | 构造空的栈 |
empty() | 检测 stack 是否为空 |
size() | 返回 stack 中元素的个数 |
top() | 返回栈顶元素的引用 |
push() | 将元素 val 压入 stack 中 |
pop() | 将 stack 中尾部的元素d出 |
#include
#include
using namespace std;
void test_stack()
{
stack<int> st;
st.push(1);
st.push(2);
st.push(3);
while(!st.empty())
{
cout << st.top() << " ";
st.pop();
}
cout << endl;
}
int main()
{
test_stack();
return 0;
}
💦 stack的OJ
1、最小栈<难度系数⭐>
📝 题述:设计一个支持 push ,pop ,top *** 作,并能在常数时间内检索到最小元素的栈。
实现 MinStack 类(要求以下接口的时间复杂度都是 O(1)):
- MinStack() 初始化堆栈对象。
- void push(int val) 将元素 val 推入堆栈。
- void pop() 删除堆栈顶部的元素。
- int top() 获取堆栈顶部的元素。
- int getMin() 获取堆栈中的最小元素。
💨示例1:
输入:
[“MinStack”,“push”,“push”,“push”,“getMin”,“pop”,“top”,“getMin”]
[[],[-2],[0],[-3],[],[],[],[]]
输出:
[null,null,null,null,-3,null,0,-2]
解释:
MinStack minStack = new MinStack();
minStack.push(-2);
minStack.push(0);
minStack.push(-3);
minStack.getMin(); -> 返回 -3.
minStack.pop();
minStack.top(); -> 返回 0.
minStack.getMin(); -> 返回 -2.
⚠ 提示:
- -231 <= val <= 231 - 1
- pop、top 和 getMin *** 作总是在非空栈上调用
- push、pop、top、and getMin 最多被调用 3 * 104 次
🧷 平台:Visual studio 2017 && windows
🔑 核心思想:这里它并没有要求我们使用数组或链表去原生实现,我们这里使用库里的栈,实现接口功能即可。
比如【3, 8, 5, 0 (这里先 push 4 个数据,再 pop 一个数据)】,其次定义 _min 去记录最小值,每次 push 满足条件时就更新 _min,但是当 pop 时就会把 _min 的值删除掉,这时的最小值是 3,但是你怎么写才能知道是 3,你必须得遍历一遍栈里的所有数据,才能知道最小值是 3,而此时的 pop 就不再是 O(1) 了。
所以我们正确的 *** 作应该给两个栈,一个栈存正常值,另一个栈存最小值(注意这里的最小值存多个),比如【3, 8, 5, 0 (这里先 push 4 个数据,再 pop 一个数据)】,这里在往第一个栈 push 时就记录最小值到第二个栈【3, 0】,如果两个栈里的值 pop 是一样的,那就都 pop,【3】,否则就只 pop 第一个栈。这就是经典的以空间换时间的思想。
边缘问题,比如【3, 8, 5, 0 (这里先 push 4 个数据,再 pop 一个数据,再 push 0,4,0)】,最后一个 0 需要 push 吗 ? 答案是需要的,如果不 push,再 pop 的话就会把最小值给删除(因为这里栈顶的数据是相同的),此时 getMin 就是 5,但是其实不是。
leetcode原题
class MinStack {
public:
//这里其实可以不用写它的构造函数(把它删了也ok),因为_st and _minst都是自定义类型(调用默认构造初始化),同时也不需要实现析构函数(调用默认析构(栈的析构)),同理拷贝构造和赋值也不需要。
MinStack() {
}
void push(int val) {
_st.push(val);
//更新栈
if(_minst.empty() || val <= _minst.top())
{
_minst.push(val);
}
}
void pop() {
//_st必须pop,相同就都pop
if(_st.top() == _minst.top())
{
_minst.pop();
}
_st.pop();
}
int top() {
return _st.top();
}
int getMin() {
//_minist的栈顶就是当前_st的最小值
return _minst.top();
}
stack<int> _st;
stack<int> _minst;
};
/**
* Your MinStack object will be instantiated and called as such:
* MinStack* obj = new MinStack();
* obj->push(val);
* obj->pop();
* int param_3 = obj->top();
* int param_4 = obj->getMin();
*/
2、栈的d出压入序列<难度系数⭐⭐>
📝 题述:输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的d出顺序。假设压入栈的所有数字均不相等。例如序列 1,2,3,4,5 是某栈的压入顺序,序列 4,5,3,2,1 是该压栈序列对应的一个d出序列,但 4,5,3,2,1 就不可能是该压栈序列的d出序列。
⚠ 提示:
- 0 <= pushV.length == popV.length <= 1000
- -1000 <= pushV[i] <= 1000
- pushV 的所有数字均不相同
💨示例1:
输入:
[1,2,3,4,5],[4,5,3,2,1]
返回值:
true
说明:
可以通过
push(1) => push(2) => push(3) => push(4) => pop() => push(5) => pop() => pop() => pop() => pop()
这样的顺序得到 [4,5,3,2,1] 这个序列,返回 true。
💨示例2:
输入:
[1,2,3,4,5],[4,3,5,1,2]
返回值:
false
说明:
由于是 [1,2,3,4,5] 的压入顺序,[4,3,5,1,2] 的d出顺序,要求 4,3,5 必须在 1,2 前压入,且 1,2 不能d出,但是这样压入的顺序,1 又不能在 2 之前d出,所以无法形成,返回 false。
🧷 平台:Visual studio 2017 && windows
🔑 核心思想:这道题之前我们有碰到过选择题。这道题本质就是模拟栈的特性 “Last In First Out”。
这里定义了一个栈来模拟,不管三七二十一,pushi 先入栈,随后 ++,出栈的顺序一定是入栈后再出的,所以每次入栈后都需要判断 pushi and popi 是否相等,相等就出(且要循环着出),否则就入,它们两个都能走到最后,就说明是匹配的。
nowcoder原题
class Solution {
public:
bool IsPopOrder(vector<int> pushV,vector<int> popV) {
stack<int> st;
size_t pushi = 0, popi = 0;
//压入顺序结束就必须出结果
while(pushi < pushV.size())
{
//先入一个数据,然后++
st.push(pushV[pushi++]);
//循环出栈
while(!st.empty() && st.top() == popV[popi])
{
++popi;
st.pop();
}
}
//st为空,说明匹配
return st.empty();
//同上
//return popi == popV.size();
}
};
3、逆波兰表达式求值<难度系数⭐⭐>
📝 题述:根据 逆波兰表示法,求表达式的值。有效的算符包括 +、-、*、/ 。每个运算对象可以是整数,也可以是另一个逆波兰表达式。注意 两个整数之间的除法只保留整数部分。可以保证给定的逆波兰表达式总是有效的。换句话说,表达式总会得出有效数值且不存在除数为 0 的情况。
💨示例1:
输入:tokens = [“2”,“1”,“+”,“3”,“*”]
输出:9
解释:该算式转化为常见的中缀算术表达式为:((2 + 1) * 3) = 9
💨示例2:
输入:tokens = [“4”,“13”,“5”,“/”,“+”]
输出:6
解释:该算式转化为常见的中缀算术表达式为:(4 + (13 / 5)) = 6
💨示例3:
输入:tokens = [“10”,“6”,“9”,“3”,“+”,“-11”,““,”/“,””,“17”,“+”,“5”,“+”]
输出:22
解释:该算式转化为常见的中缀算术表达式为:
((10 * (6 / ((9 + 3) * -11))) + 17) + 5
= ((10 * (6 / (12 * -11))) + 17) + 5
= ((10 * (6 / -132)) + 17) + 5
= ((10 * 0) + 17) + 5
= (0 + 17) + 5
= 17 + 5
= 22
⚠ 提示:
- 1 <= tokens.length <= 104
- tokens[i] 是一个算符(“+”、“-”、“*” 或 “/”),或是在范围 [-200, 200] 内的一个整数
逆波兰表达式:
逆波兰表达式是一种后缀表达式,所谓后缀就是指算符写在后面。
-
平常使用的算式则是一种中缀表达式,如 ( 1 + 2 ) * ( 3 + 4 ) 。
中缀表达式面临的最大问题在于程序不方便运算,因为运算符优先级的问题,所以我们处理这种问题可以先将中缀表达式转换成后缀表达式,然后用后缀表达式进行运算。
大概了解下中缀表达式转后缀表达式,这里需要借助栈
-
该算式的逆波兰表达式写法为 ( ( 1 2 + ) ( 3 4 + ) * ) 。
逆波兰表达式主要有以下两个优点:
- 去掉括号后表达式无歧义,上式即便写成 1 2 + 3 4 + * 也可以依据次序计算出正确结果。
- 适合用栈 *** 作运算:遇到数字则入栈;遇到算符则取出栈顶两个数字进行计算,并将结果压入栈中
🧷 平台:Visual studio 2017 && windows
🔑 核心思想:在了解完后缀表达式怎么由中缀表达式转换后,这里题目本意是需要我们计算后缀表达式的值。
leetcode原题
🧿 版本一
class Solution {
public:
int evalRPN(vector<string>& tokens) {
stack<int> st;
for(const auto str : tokens)
{
//建议不要这样写,因为如果 *** 作数是负数,就会出bug
/*switch(str[0])
{
case: '+':
//... ...
}*/
int left, right;
//+、-、*、/,就出两个栈顶的元素,top1对应right,top2对应left,再把计算的结果入栈
if(str == "+")
{
right = st.top();
st.pop();
left = st.top();
st.pop();
st.push(left + right);
}
else if(str == "-")
{
right = st.top();
st.pop();
left = st.top();
st.pop();
st.push(left - right);
}
else if(str == "*")
{
right = st.top();
st.pop();
left = st.top();
st.pop();
st.push(left * right);
}
else if(str == "/")
{
right = st.top();
st.pop();
left = st.top();
st.pop();
st.push(left / right);
}
else// *** 作数
{
//入栈前,将字符串转整型
st.push(stoi(str));
}
}
//返回此时栈顶的元素
return st.top();
}
};
🧿 版本二 (优化版本一)
优化的点在于 “ 在判断 *** 作符时有大量冗余的代码 ”。
解决方案:
- 封装一个成员函数 (能解决,但还能更好的方法 ?)。
- 这道题使用 map 非常简单,但目前我们还没学,就不谈了。
- 使用逻辑或 “ || ”,这种写法的问题是把运算结果 push 时不知道是什么 *** 作符。解决方法就是定义一个 48 大小的数组建立映射关系,比如在下标 47 的位置存储 “ / ”,然后根据对应的字符就可以取到对应的符号,但是数组里所存储的字符,不是类型,所以没错,~~ 翻车了,连第 2 种方案好像也翻车了,所以这里给成员函数好像是比较好的方案了,或者在 “ || ” 的基础上使用 switch 语句 (这两种方法差不多,只是减少了代码量,本质并没有多少的改进)。无妨,多翻车才能更好的上车嘛 !!!
注:其实也有更好的简化的方案的,只不过目前我们玩不了,这里先吊下大家的胃口 —— C++11 的包装器。也欢迎大家有更好的方案可以在评论区留言。
//版本二(优化版本一)
class Solution {
public:
//解决方案一:
void getnum(stack<int>& st, int& l, int& r)
{}
int evalRPN(vector<string>& tokens) {
stack<int> st;
for(const auto str : tokens)
{
int left, right;
if(str == "+" || str == "-" || str == "*" || str == "/")
{
right = st.top();
st.pop();
left = st.top();
st.pop();
switch(str[0])
{
case '+':
st.push(left + right);
break;
case '-':
st.push(left - right);
break;
case '*':
st.push(left * right);
break;
case '/':
st.push(left / right);
break;
}
}
else
{
st.push(stoi(str));
}
}
return st.top();
}
};
🧿 版本三 (优化版本二,骚 *** 作)
这里可以先跳过,把后面 C++11 的包装器、map 等,等学了再来看。
class Solution {
public:
int evalRPN(vector<string>& tokens) {
map<string, function<int(int, int)>> opCountMap =
{
{"+", [](int x, int y)->int{return x + y;}},
{"-", [](int x, int y)->int{return x - y;}},
{"*", [](int x, int y)->int{return x * y;}},
{"/", [](int x, int y)->int{return x / y;}}
};
stack<int> st;
for(auto& str : tokens)
{
if(str == "+" || str == "-" || str == "*" || str == "/")
{
int right = st.top();
st.pop();
int left = st.top();
st.pop();
st.push(opCountMap[str] (left, right));
}
else
{
st.push(stoi(str));
}
}
return st.top();
}
};
4、用栈实现队列<难度系数⭐>
📝 题述:请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有 *** 作 (push、pop、peek、empty):
实现 MyQueue 类:
- void push(int x) 将元素 x 推到队列的末尾。
- int pop() 从队列的开头移除并返回元素。
- int peek() 返回队列开头的元素。
- boolean empty() 如果队列为空,返回 true;否则,返回 false。
⚠ 说明:
- 你只能使用标准的栈 *** 作 —— 也就是只有 push to top、peek/pop from top、size、is empty *** 作是合法的。
- 你所使用的语言也许不支持栈。你可以使用 list 或者 deque (双端队列) 来模拟一个栈,只要是标准的栈 *** 作即可。
💨示例1:
输入:
[“MyQueue”, “push”, “push”, “peek”, “pop”, “empty”]
[[], [1], [2], [], [], []]
输出:
[null, null, null, 1, 1, false]
解释:
MyQueue myQueue = new MyQueue();
myQueue.push(1); // queue is: [1]
myQueue.push(2); // queue is: [1, 2] (leftmost is front of the queue)
myQueue.peek(); // return 1
myQueue.pop(); // return 1, queue is [2]
myQueue.empty(); // return false
⚠ 提示:
- 1 <= x <= 9
- 最多调用 100 次 push、pop、peek 和 empty
- 假设所有 *** 作都是有效的 (例如,一个空的队列不会调用 pop 或者 peek *** 作)
☣ 进阶:
你能否实现每个 *** 作均摊时间复杂度为 O(1) 的队列 ?换句话说,执行 n 个 *** 作的总时间复杂度为 O(n) ,即使其中一个 *** 作可能花费较长时间。
🧷 平台:Visual studio 2017 && windows
🔑 核心思想:我们之前用 C语言写过 “ 两个队列实现栈 ” and “ 两个栈实现队列 ”。用 C++ 实现就很简单了。
leetcode原题
4、用队列实现栈<难度系数⭐>
leetcode原题
💦 stack的模拟实现vector 模拟实现 stack。
#include
namespace bit
{
template<class T>
class stack
{
public:
stack(){}
//先进
void push(const T& x){_ve.push_back(x);}
//后出
void pop(){_ve.pop_back();}
const T& top(){return _ve.back();}
size_t size(){return _ve.size();}
bool empty(){return _ve.empty();}
private:
std::vector<T> _ve;
};
}
二、queue的介绍及使用
💦 queue的介绍
queue文档介绍
-
队列是一种容器适配器,专门用于在 FIFO 上下文(先进先出)中 *** 作,其中从容器一端插入元素,另一端提取元素。
-
队列作为容器适配器实现,容器适配器即将特定容器类封装作为其底层容器类,queue 提供一组特定的成员函数来访问其元素。元素从队尾入队列,从队头出队列。
-
底层容器可以是标准容器类模板之一,也可以是其他专门设计的容器类。该底层容器应至少支持以下 *** 作:
➡ empty:检测队列是否为空
➡ size:返回队列中有效元素的个数
➡ front:返回队头元素的引用
➡ back:返回队尾元素的引用
➡ push_back:在队列尾部入队列
➡ pop_front:在队列头部出队列 -
标准容器类 deque 和 list 满足了这些要求。默认情况下,如果没有为 queue 实例化指定容器类,则使用标准容器 deque。
函数声明 | 接口说明 |
---|---|
queue() | 构造空的队列 |
empty() | 检测队列是否为空,是返回 true,否则返回 false |
size() | 返回队列中有效元素的个数 |
front() | 返回队头元素的引用 |
back() | 返回队尾元素的引用 |
push() | 在队尾将元素 val 入队列 |
pop() | 将队头元素出队列 |
#include
#include
using namespace std;
void test_queue()
{
queue<int> q;
q.push(1);
q.push(2);
q.push(3);
while(!q.empty())
{
//queue与stack相同的是入数据都是push,但出数据stack是top,queue是front
cout << q.front() << " ";
q.pop();
}
cout << endl;
}
int main()
{
test_queue();
return 0;
}
💦 queue的模拟实现
list 模拟实现 queue。
#include
namespace bit
{
template class<T>
class queue
{
public:
queue(){}
//先进
void push(const T& x){_qu.push_back(x);}
//先出
void pop(){_qu.pop_front();}
const T& front(){return _qu.front();}
size_t size(){return _qu.size();}
bool empty(){return _qu.empty();}
private:
std::list<T> _qu;
};
}
三、priority_queue的介绍及使用
💦 priority_queue的介绍
priority_queue文档介绍
-
优先队列是一种容器适配器,根据严格的弱排序标准,它的第一个元素总是它所包含的元素中最大的。
-
此上下文类似于堆,在堆中可以随时插入元素,并且只能检索最大堆元素 (优先队列中位于顶部的元素)。
-
优先队列被实现为容器适配器,容器适配器即将特定容器类封装作为其底层容器类,queue 提供一组特定的成员函数来访问其元素。元素从特定容器的 “ 尾部 ” d出,其称为优先队列的顶部。
-
底层容器可以是任何标准容器类模板,也可以是其他特定设计的容器类。容器应该可以通过随机访问迭代器访问,并支持以下 *** 作:
➡ empty():检测容器是否为空。
➡ size():返回容器中有效元素个数。
➡ front():返回容器中第一个元素的引用。
➡ push_back():在容器尾部插入元素。
➡ pop_back():删除容器尾部元素。 -
标准容器类 vector 和 deque 满足这些需求。默认情况下,如果没有为特定的 priority_queue 类实例化指定容器类,则使用 vector。
-
需要支持随机访问迭代器,以便始终在内部保持堆结构。容器适配器通过在需要时自动调用算法函数 make_heap、push_heap 和pop_heap 来自动完成此 *** 作。
注意这里的优先级队列不符合队列的先进先出特性,它的 push 没有要求,pop/top 是取优先级最高的,优先级指的是大小,这里默认是大的优先级高,如果要让小的优先级高,就需要仿函数来控制。但要注意并不一定是大的优先级高,小的优先级低,因为对于用数字来评分 (大的高) 是一种场景,对于用字母来评分 (小的高) 又是一种场景。
💦 priority_queue的使用优先级队列默认使用 vector 作为其底层存储数据的容器,在 vector 上又使用了堆算法将 vector 中元素构造成堆的结构,因此 priority_queue 就是堆,所有需要用到堆的位置,都可以考虑使用 priority_queue。注意:默认情况下 priority_queue 是大堆。
函数声明 | 接口说明 |
---|---|
priority_queue()/priority_queue(first,last) | 构造一个空的优先级队列 |
empty() | 检测优先级队列是否为空,是返回 true,否则返回 false |
top() | 返回优先级队列中最大 (最小) 元素,即堆顶元素 |
push(x) | 在优先级队列中插入元素 x |
pop() | 删除优先级队列中最大 (最小) 元素,即堆顶元素 |
#include
#include
#include //greater的头
using namespace std;
void test_priority_queue()
{
//priority_queue pq;//默认是大堆,大的优先级高
priority_queue<int,vector<int>, greater<int>> pq;//默认是小堆,小的优先级高。控制大小堆的是第3个参数,你要传第3个参数,必须先传第2个参数,因为它也是缺省的
pq.push(1);
pq.push(10);
pq.push(11);
pq.push(3);
pq.push(5);
pq.push(8);
while(!pq.empty())
{
//取堆顶的数据
cout << pq.top() << " ";
//出堆顶(与最后元素交换,再删除它),向下调整
pq.pop();
}
cout << endl;
}
int main()
{
test_priority_queue();
return 0;
}
💦 priority_queue的OJ
1、数组中的第K个最大元素<难度系数⭐⭐>
📝 题述:给定整数数组 nums 和整数 k,请返回数组中第 k 个最大的元素。请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。
💨示例1:
输入:[3,2,1,5,6,4] 和 k = 2
输出:5
💨示例2:
输入:[3,2,3,1,2,4,5,5,6] 和 k = 4
输出:4
⚠ 提示:
- 1 <= k <= nums.length <= 104
- -104 <= nums[i] <= 104
🧷 平台:Visual studio 2017 && windows
🔑 核心思想:这道题是一道 top-k 的变形问题。这里用 C++ 来做就非常简单,有如下方案:
-
先调用 中的 sort 对数组排升序,随后算倒数第 k 大就是正数的 nums.size() - k 为下标的位置。当然直接排降序也可以,其底层是快排。时间复杂度为 O(N*logN)。无空间复杂度。
-
用 priority_queue 建一个大堆,然后pop k - 1 个,再 top() 就是第 k 大的。把 nums 的数据往 maxHeap 里放有 2 种方式。时间复杂度为 O(N + K * logN),N 是向下调整建堆,K * logN 是 pop。空间复杂度 O(N)。
➡ 遍历nums,一个个 push。
➡ 构造函数里提供迭代器区间初始化。 -
建一个 k 个数的小堆,其它数依次比较,它比堆顶的数要大,那么就 pop 堆顶,再 push 这个数,最后堆顶的数据就是第 k 大的。时间复杂度是 O(K + (N - K) * logK),空间复杂度是 O(K)。就目前的数据量来说,其实第二种方案和第三种方案的效率差不多,但是当 N 远大于 K 时,尤其是大到内存存不下 (方案一、二都不能适用),第三种方案更适用,此时可以认为它的时间复杂度是 O(N - K) ➡ O(N)。
leetcode原题
🧿 方案一
class Solution {
public:
int findKthLargest(vector<int>& nums, int k) {
// //RandAccessIterator,sort默认是升序
// sort(nums.begin(), nums.end());
// //此时倒数第k大就是正数的nums.size()-k为下标的位置。
// return nums[nums.size() - k];
//sort排降序
//greater g;
//sort(nums.begin(), nums.end(), g);
sort(nums.begin(), nums.end(), greater<int>());//同上,匿名对象
//此时k-1作为下标就是第k大的
return nums[k - 1];
}
};
🧿 方案二
class Solution {
public:
int findKthLargest(vector<int>& nums, int k) {
priority_queue<int> maxHeap(nums.begin(), nums.end());
while(k-=1)//同k-=1
{
maxHeap.pop();
}
//此时堆顶既是第k大
return maxHeap.top();
}
};
🧿 方案三
class Solution {
public:
int findKthLargest(vector<int>& nums, int k) {
//建k个数的小堆
priority_queue<int, vector<int>, greater<int>> kMinHeap;
for(size_t i = 0; i < k; ++i)
{
kMinHeap.push(nums[i]);
}
//再用剩余的数比较
for(size_t j = k; j < nums.size(); ++j)
{
if(kMinHeap.top() < nums[j])
{
kMinHeap.pop();
kMinHeap.push(nums[j]);
}
}
//此时的堆顶就是第k大的
return kMinHeap.top();
}
};
💦 priority_queue的模拟实现
💨PriorityQueue.h
#pragma once
#include
#include
#include
using namespace std;
namespace bit
{
//仿函数/函数对象——自定义类型,类型的对象可以像函数一样使用
template<class T>
class Less
{
public:
bool operator()(const T& x, const T& y)
{
return x < y;
}
};
template<class T>
class Greater
{
public:
bool operator()(const T& x, const T& y)
{
return x > y;
}
};
//模板参数->类型
//函数参数->对象
//less->大堆 greater->小堆,默认是Less
template<class T, class Container = vector<T>, class Compare = Less<T>>
class priority_queue
{
public:
priority_queue()
{}
template<class InputIterator>
priority_queue(InputIterator first, InputIterator last)//可以传任意类型的迭代器
{
//插入数据
while(first != last)
{
_con.push_back(*first);
++first;
}
//建堆
for(int i = (_con.size() - 1 - 1) / 2; i >= 0; --i)
{
AdjustDown(i);
}
}
void AdjustUp(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[parent], _con[child]);
child = parent;
parent = (child - 1) / 2;
}
else//父亲>=孩子
{
break;
}
}
}
void AdjustDown(size_t parent)
{
Compare com;
size_t child = parent * 2 + 1;
while (child < _con.size())
{
//if (child + 1 < _con.size() && _con[child] < _con[child + 1])
if (child + 1 < _con.size() && com(_con[child], _con[child + 1]))
{
++child;
}
//if (_con[parent] < _con[child])
if (com(_con[parent], _con[child]))
{
swap(_con[parent], _con[child]);
parent = child;
child = parent * 2 + 1;
}
else
{
break;
}
}
}
void push(const T& x)
{
_con.push_back(x);
AdjustUp(_con.size() - 1);
}
void pop()
{
swap(_con[0], _con[_con.size() - 1]);
_con.pop_back();
AdjustDown(0);
}
const T& top() const
{
return _con[0];
}
size_t size()
{
return _con.size();
}
bool empty()
{
return _con.empty();
}
private:
Container _con;//构造析构等可以不写,因为这是自定义类型
};
void test_priority_queue()
{
//priority_queue pq;
priority_queue<int, vector<int>, Greater<int>> pq;
pq.push(1);
pq.push(10);
pq.push(11);
pq.push(3);
pq.push(5);
pq.push(8);
while (!pq.empty())
{
cout << pq.top() << " ";
pq.pop();
}
cout << endl;
list<int> lt = { 2, 5, 1, 3, 4 };
priority_queue<int, vector<int>, greater<int>> pq1(lt.begin(), lt.end());
while (!pq1.empty())
{
cout << pq1.top() << " ";
pq1.pop();
}
cout << endl;
}
}
💨test.cpp
int main()
{
bit::test_priority_queue();
return 0;
}
📝说明
-
List item
在以前我们要让大堆变小堆,都是直接去改符号,那如何能不改符号的情况下就能达到效果呢 ❓
C语言其实可以利用函数指针来解决。但是 C++ 放弃用函数指针的方式,且非常不建议用函数指针,因为比较复杂。可以说 C++ 里的仿函数/函数对象就是为了替代 C语言里的函数指针。
“ () ” 的功能可以提高优先级、强制类型转换、函数名 (形参表),仿函数用的是函数名 (形参表)。我们实现两个类,分别重载 “ () ” 完成比较大小功能,然后用 priority_queue 这个类来实例化控制。
-
List item
对比 priority_queue 的仿函数和 中 sort 的仿函数 ❗
priority_queue 是一个类,最好通过模板参数来传递,传的是类型。
这里 less 中给的是 typename Container::value_type,并没有很特别的原因,less 里可以直接给 T,因为它是取 Container 中的 value_type,value_type 是被 typedef 的第一个模板参数,也就是 T。我们要去 Container 中取 value_type。Container 是一个模板,不知道它具体是啥,因为 vector 没有被实例化,所以去取可能会报错,所以加上 typename 是告诉编译器,后面是一个类型,当然报错与否,主要看编译器。这里后面遇到合适的场景会再谈。
sort 是一个函数,最好通过参数来传递。
类模板是显示实例化的,所以说我们可以显示的指定参数;但是函数模板一般是通过实参去推演的,所以说 sort 得写在函数参数里。所以 priority_queue 是传类型,sort 是传对象。
要求往优先级队列里放一个日期类,且取出最大的日期 ❓
🧿 方案一:
#include
#include
using namespace std;
class Date
{
public:
Date(int year = 2022, int month = 5, int day = 1)
: _year(year)
, _month(month)
, _day(day)
{}
bool operator<(const Date& d)const
{
return (_year < d._year) ||
(_year == d._year && _month < d._month) ||
(_year == d._year && _month == d._month && _day < d._day);
}
friend ostream& operator<<(ostream& _cout, const Date& d);
private:
int _year;
int _month;
int _day;
};
ostream& operator<<(ostream& _cout, const Date& d)
{
_cout << d._year << "-" << d._month << "-" << d._day << endl;
return _cout;
}
int main()
{
//我们可以放int,也可以放类
priority_queue<Date> pq;
pq.push(Date(2021, 11, 24));
pq.push(Date(2021, 10, 24));
pq.push(Date(2021, 12, 24));
pq.push(Date(2022, 1, 24));
//pq是自定义类型,所以上面需要重载<<,要取最大的日期,需要重载<
cout << pq.top() << endl;
pq.pop();
return 0;
}
🧿 方案二:
假设极端情况,如果存的是日期类指针呢,如果再用上面的方式,取不出来,因为它比的是地址。解决方案就是我们自己实现仿函数来控制。
#include
#include
using namespace std;
class Date
{
public:
Date(int year = 2022, int month = 5, int day = 1)
: _year(year)
, _month(month)
, _day(day)
{}
bool operator<(const Date& d)const
{
return (_year < d._year) ||
(_year == d._year && _month < d._month) ||
(_year == d._year && _month == d._month && _day < d._day);
}
friend ostream& operator<<(ostream& _cout, const Date& d);
friend class PDateLess;
private:
int _year;
int _month;
int _day;
};
ostream& operator<<(ostream& _cout, const Date& d)
{
_cout << d._year << "-" << d._month << "-" << d._day << endl;
return _cout;
}
class PDateLess
{
public:
bool operator()(const Date* p1, const Date* p2)
{
/*if(p1->_year < p2->_year)
|| (p1->_year == p2->_year && p1->_month < p2->_month)
|| (p1->_year == p2->_year && p1->_month == p2->_month && p1->_day < p2->_day)
{
return true;
}
else
{
return false;
}*/
//同上,重载了<的情况
return *p1 < *p2;
}
};
int main()
{
priority_queue<Date*, vector<Date*>, PDateLess> pq;
pq.push(new Date(2023, 11, 24));
pq.push(new Date(2021, 10, 24));
pq.push(new Date(2021, 12, 24));
pq.push(new Date(2022, 1, 24));
cout << (*pq.top()) << endl;
pq.pop();
return 0;
}
最后这里想说明的是我们可以通过仿函数来控制比较方式。
四、容器适配器 💦 什么是适配器适配器是一种设计模式 (设计模式是一套被反复使用的、多数人知晓的、经过分类编目的、代码设计经验的总结),该种模式是将一个类的接口转换成客户希望的另外一个接口。
虽然 stack 和 queue 中也可以存放元素,但在 STL 中并没有将其划分在容器的行列,而是将其称为容器适配器,这是因为 stack 和队列只是对其他容器的接口进行了包装,STL 中 stack 和 queue 默认使用 deque,比如:
deque文档介绍
这里我们只是了解了解,想更深入可以去配合着源码看候捷老师的《STL源码剖析》。
1、deque的简单使用void test_deque()
{
deque<int> dq;
dq.push_back(1);
dq.push_back(2);
dq.push_front(3);
dq.push_front(4);
//dq.insert();
//dq.erase();
dq.pop_back();
dq.pop_front()
for(size_t i = 0; i < dq.size(); ++i)
{
cout << dq[i] << " ";
}
cout << endl;
}
📝说明
通过以上 *** 作我们可以看出 deque 融合了 vector and list 的优点,并且从使用的角度避开了 vector and list 各自的缺点。
- vector 的缺点是需要扩容,且头部和中间的插入删除。
- list 的缺点是 “ [ ] ”,它不支持任意位置的随机访问。
那就目前来看 deque 可以认为是最完美的线性表。那我们直接学 deque 就行了,还学啥 vector and list 呢 ? 实际 deque 并没有崭露头角,想要了解从使用方面 deque 这么牛逼,却没有把 vector and list 替换掉,就需要了解 deque 的底层实现(这里我们也只是了解,不会模拟实现)。
2、deque的原理介绍deque(双端队列):但是记住,它与实际队列的先进先出的性质没有关联,也就是说 deque 并不要求先进先出。它是一种双开口的 " 连续 " 空间的数据结构,双开口的含义是:可以在头尾两端进行插入和删除 *** 作,且时间复杂度为 O(1),与 vector 比较,头插效率高,不需要搬移元素;与 list 比较,空间利用率比较高。
我们要继续分析 deque 就要先分析 vector and list,因为开发者肯定是从它们俩上受启发的。
- vector 是连续的物理空间,它的优点是 a)尾插、尾删效率很高,最重要的是它支持随机访问 b)cpu高速缓存命中率很高。它的缺点是 a)空间不够,就需要增容,增容代价大,还存在一定的空间浪费。 b)头部和中间的插入删除是 O(N),效率低。
- list 是非连续的物理空间,它的优点是 a)按需申请释放空间。b)任意位置插入删除都是 O(1),效率高。它的缺点是 a)不支持随机访问。b)cpu高速缓存命中率低。
可以看到 vector and list 的优缺点是互补的,vector 的优点大概就是 list 的缺点,vector 的缺点大概是 list 的优点,list 的优点大概就是 vector 的缺点,list 的缺点大概是 vector 的优点。
结合 vector and list 的优缺点,开发者就设计了 deque(但这似乎在很多场景下是一个相对比较无用的设计)。
deque 并不是真正连续的空间,而是由一段段连续的小空间拼接而成的,实际 deque 类似于一个动态的二维数组,其底层结构如下图所示:
deque 设计的真正复杂的是它的迭代器,这也是它的核心。我们碰到的迭代器要么是原生指针,要么是类封装的指针。而 deque 的迭代器封装了 4 个指针。
双端队列底层是一段假象的连续空间,实际是分段连续的,为了维护其 “ 整体连续 ” 以及随机访问的假象,任务就落在了 deque 的迭代器身上,因此 deque 的迭代器设计就比较复杂,如下图所示:
📝说明:
- 它的迭代器有 4 个指针,其中 first and last 分别指向一段 buff 的开始和结束,cur 指向当前迭代器所访问的位置,node 指向中控数组。
- 遍历:begin() 给了第一个 buff 里面,*cur 就是第一个数据,++cur 就指向第二个数据,cur == last 时,第一个 buff 走完,随后 node = *node++,把 first and last 定位到下一个 buff 的对应位置,然后接着遍历即可。
以上内容在时间有富余的情况下可以在深挖。不过有人也遇到过相关面试题。
【面试题】
结合 vector and list 的优缺点,有没有什么改进的方案 ❓
这里呢,不要一上嘴就谈 deque 迭代器的 4 个指针啥的,你可以先不需要谈细节的实现。就谈可以实现固定数组的 buff,头插、尾插,再通过中控数组也就是指针数组来管理 buff,如果中控满了,扩容中控。在没有良好功底的情况下,面试官不问迭代器的实现就不谈,避免踩坑,因为往下会把这个问题扯的很深。
2、deque的缺陷及优点咦,上面 deque 不是挺牛逼的吗 ?
deque 叫双端队列,说明它很适合头插头删,尾插尾删,也说明 deque 很适合 (比 vector and list 都适合) 去做 stack and queue 的默认适配容器,它尾插数据相比 vector 没有扩容,相比 list 它的缓存利用率很高。所以说 deque 最大的优点是做 stack and queue 的默认适配容器。
deque 的一大缺陷就是中间插入、删除,效率很低,比如要删除一个数据,这里有两种实现方案:
- 可以头往前挪,可以尾往前挪,也可以哪一端少就挪哪一端,这又回到 vector 的问题了。
- 只挪当前 buff 的数据,记录当前 buff 的 size(可以定义结构体,它里面放中控数组及 size),少一个就缩容,多一个就增容,但它始终都要挪动数据。且当 buff 的大小不是固定后,“ [ ] ” 就不好计算了。
整体而言,第一种,中间插入删除的效率就和 vector 一样了,而 " [ ] " 的效率好歹高点;第二种,中间插入删除的效率变高了,而 " [ ] " 的效率变低了。所以对于 deque 中间插入删除 *** 作类似于一碗水端不平的问题。大家可以去瞅下 SGI 版本的 deque 是怎么实现中间插入删除的。
deque 的另一个缺陷是不够极致, deque 是一种折中的方案设计。随机访问效率不及 vector,任意位置插入删除不及 list,它真正的优点是头尾插入删除。你可以理解 vector 是一面很锋利的刀,list 也是一面很锋利的刀,所以铁匠就设计铸造出一把两面都很锋利的刀,这就导致了两面都不够锋利,它是一种不够极致,折中的设计。所以严格来说 deque 是一个挺失败的设计。
使用场景 ❓
- 数据排序,使用 vector。
- 任意位置插入、删除,使用 list。
- 头尾插入删除 (stack and queue) 用 deque。
也就是说 stack 为啥不用 vector 作为默认容器;queue 为啥不用 list 做为默认容器 ❓
stack 是一种后进先出的特殊线性数据结构,因此只要具有 push_back and pop_back *** 作的线性结构,都可以作为 stack 的底层容器,比如 vector and list 都可以;queue 是先进先出的特殊线性数据结构,只要具有 push_back and pop_front *** 作的线性结构,都可以作为 queue 的底层容器,比如 list。但是 STL 中对 stack and queue 默认选择 deque 作为其底层容器,主要是因为:
- stack 和 queue 不需要遍历 (因此 stack and queue没有迭代器),只需要在固定的一端或者两端进行 *** 作。
- 在 stack 中空间增长时,deque 比 vector 的效率高 (扩容时不需要搬移大量数据);queue 中的元素增长时,deque 不仅效率高,而且内存使用率高。
结合了 deque 的优点,而完美的避开了其缺陷。
💦 STL标准库中对于stack和queue的模拟实现 1、stack的模拟实现#include
#include
#include
#include
using namespace std;
namespace bit
{
//stack是一个Container适配(封装转换)出来的,且它是一个缺省参数,deque容器,但是它的功能是不变的,所以叫做容器适配器
template<class T, class Container = deque<T>>
class stack
{
//Container的尾认定是栈顶
public:
void push(const T& x)//先进
{
_con.push_back(x);
}
void pop()//后出
{
_con.pop_back();
}
const T& top()
{
//不能用[],因为如果是list就不支持了,back更为通用
return _con.back();
}
size_t size()
{
return _con.size();
}
bool empty()
{
return _con.empty();
}
private:
Container _con;
};
void test_stack()
{
//stack> st;
//stack> st;
stack<int> st;
st.push(1);
st.push(2);
st.push(3);
while(!st.empty())
{
cout << st.top() << " ";
st.pop();
}
cout << endl;
}
}
int main()
{
bit::test_stack();
return 0;
}
2、queue的模拟实现
#include
#include
#include
#include
using namespace std;
namespace bit
{
//stack是一个Container适配(封装转换)出来的,且它是一个缺省参数,deque容器
template<class T, class Container = deque<T>>
class queue
{
//Container的尾认定是队尾,头是队头
public:
void push(const T& x)//先进
{
_con.push_back(x);
}
void pop()//先出
{
_con.pop_front();
}
const T& front()
{
return _con.front();
}
const T& back()
{
return _con.back();
}
size_t size()
{
return _con.size();
}
bool empty()
{
return _con.empty();
}
private:
Container _con;
};
void test_queue()
{
//queue> qu;//err,因为vector不支持头删接口,所以不能适配
//queue> qu;
queue<int> qu;
qu.push(1);
qu.push(2);
qu.push(3);
while(!qu.empty())
{
cout << qu.front() << " ";
qu.pop();
}
cout << endl;
}
}
int main()
{
bit::test_queue();
return 0;
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)