Leetcode栈&队列
Leetcode栈&队列
232.用栈实现队列
题干:
思路:
- 栈是
FILO,队列是FIFO,所以如果要用栈实现队列,目的就是要栈实现一个FIFO的特性。 - 具体实现方法可以理解为,准备两个栈,一个栈用作输入栈,入数据就存数据,一个栈用作输出栈,出数据就入数据再弹数据。

代码:
class MyQueue {
/**
* 先声明变量,留在MyQueue方法中进行Init
*/
Stack<Integer> stackIn;
Stack<Integer> stackOut;
public MyQueue() {
stackIn=new Stack<Integer>();
stackOut=new Stack<Integer>();
}
public void push(int x) {
stackIn.push(x);
}
public int pop() {
dumpStackIn();
return stackOut.pop();
}
public int peek() {
dumpStackIn();
return stackOut.peek();
}
public boolean empty() {
//不难得出,当输入输出栈均为空时,队列就是空的
return stackOut.isEmpty()&&stackIn.isEmpty();
}
private void dumpStackIn(){
if(!stackOut.isEmpty()){
return;
}
while (!stackIn.isEmpty()){
stackOut.push(stackIn.pop());
}
}
}
20.有效的括号
题干:

思路:
- 这种匹配类的题目都可以用栈来解决。当某组括号的左括号被扫描到时,我们就将与它配对的括号入栈(它本身也可以,但是存入配对的括号在后续步骤中可以通过判断是否相同来加快速度),之后遇到相同的括号就将其出栈。
- 第一种情况:扫描完毕,栈同时也是空的。对应着有效的字符串
- 第二种情况:扫描未完毕,栈已经空了。对应着左括号数量少于右括号,不符合要求
- 第三种情况:扫描完毕,栈未空,说明左括号数量大于右括号,不符合要求
- 第四种情况:扫描未完毕,发现栈顶元素与当前括号类型不匹配,不符合要求
代码:
class Solution {
public boolean isValid(String s) {
//这里使用双向链表deque来模拟栈
Deque<Character> deque = new LinkedList<>();
char ch;
for (int i = 0; i < s.length(); i++) {
ch = s.charAt(i);
if (ch == '(') {
deque.push(')');
}else if (ch == '{') {
deque.push('}');
}else if (ch == '[') {
deque.push(']');
} else if (deque.isEmpty() || deque.peek() != ch) {
return false;
}else {
deque.pop();
}
}
return deque.isEmpty();
}
}
1047.删除字符串中的所有相邻重复项
题干:

思路:
- 与
有效的括号有着异曲同工之妙,我们依次扫描传入的字符串,并判断栈顶元素是否与当前扫描到的字符相同,如果相同,则出栈此元素,扫描完毕后,将栈中元素弹出逆序即可。 - 在这题中,我们将频繁使用到
删除这个操作,根据数据结构的知识,这种情况下采用Array结构将比LinkedList结构快一些
代码:
class Solution {
public String removeDuplicates(String s) {
Deque<Character> deque=new LinkedList<>();
char letter;
for(int i=0;i<s.length();i++){
letter=s.charAt(i);
if(deque.isEmpty()||deque.peek()!=letter){
deque.push(letter);
}else{
deque.pop();
}
}
String ans="";
while (!deque.isEmpty()){
//每次将上一次的ans放到末尾,就可以起到reverse的作用了
ans=deque.pop()+ans;
}
return ans;
}
}
Tips:
- 写的时候将
deque.pop()写成了deque.add(),查询代码以后发现了两者的区别 

- 首先是返回值不同,push的返回值为void,add的返回值为Boolean;其次是位置不同,push插入头,add插入尾
150.逆波兰表达式求值
题干:
Tips:
- 表达式有前缀、中缀、后缀,其中前缀表达式称为波兰表达式,这也就是为什么后缀表达式叫逆波兰表达式,中缀表达式就是人们日常运算时所书写的形式,例如
1+1+1+1=4这种。分类标准就是依据将运算符号放置在运算数前、中还是后。
思路:
- 读《算法》第四版的时候,书中的经典例题,没遇到运算符之前就入栈,遇到运算符就将运算符出栈,并同时出栈两个数字,进行与对应的出栈的运算符的运算操作,并最后将运算结果压入栈中。
- 需要注意的是,在进行减法与除法这类需要考虑顺序的运算时,不要直接就大意的认为将出栈的两个数字相减或是相除就行了。
代码:
class Solution {
public int evalRPN(String[] tokens) {
Deque<Integer> stack=new LinkedList<Integer>();
for(int i=0;i<tokens.length;i++){
if("+".equals(tokens[i])){
stack.push(stack.pop()+stack.pop());
}else if("-".equals(tokens[i])){
stack.push(-stack.pop()+stack.pop());
}else if("*".equals(tokens[i])){
stack.push(stack.pop()*stack.pop());
}else if("/".equals(tokens[i])){
int temp1=stack.pop();
int temp2=stack.pop();
stack.push(temp2/temp1);
}else{
stack.push(Integer.valueOf(tokens[i]));
}
}
return stack.pop();
}
}
347.前K个高频元素
题干:
思路:
- 第一个想法是利用哈希表存入数字与其出现的次数,排序之后取最前的K个value对应的key就可以了。
代码:
class Solution {
public int[] topKFrequent(int[] nums, int k) {
int[] ans=new int[k];
Map<Integer,Integer> map=new HashMap();
for(int i:nums){
if(map.containsKey(i)){
map.put(i,map.get(i)+1);
}else {
map.put(i,1);
}
}
int max=0;
for(Map.Entry<Integer,Integer> entry:map.entrySet()){
if(entry.getValue()>max){
max=entry.getValue();
}
}
while (k>0){
for(Map.Entry<Integer,Integer> entry:map.entrySet()){
if(entry.getValue()==max){
ans[k-1]=entry.getKey();
k--;
}
}
max--;
}
return ans;
}
}
Tips:
在Hashmap中,每一个Key-Value被看作是一个set集合。
为了更便捷的获取Key-Value,使用Entry来进行编写,for-each循环中用entrySet把每一个键值对取出进行遍历。
Leetcode栈&队列的更多相关文章
- 栈&队列&并查集&哈希表(julyedu网课整理)
date: 2018-11-25 08:31:30 updated: 2018-11-25 08:31:30 栈&队列&并查集&哈希表(julyedu网课整理) 栈和队列 1. ...
- java 集合 Connection 栈 队列 及一些常用
集合家族图 ---|Collection: 单列集合 ---|List: 有存储顺序 , 可重复 ---|ArrayList: 数组实现 , 查找快 , 增删慢 ---|LinkedList: 链表实 ...
- Java 容器之 Connection栈队列及一些常用
集合家族图 ---|Collection: 单列集合 ---|List: 有存储顺序 , 可重复 ---|ArrayList: 数组实现 , 查找快 , 增删慢 ---|LinkedList: 链表实 ...
- java面向对象的栈 队列 优先级队列的比较
栈 队列 有序队列数据结构的生命周期比那些数据库类型的结构(比如链表,树)要短得多.在程序操作执行期间他们才被创建,通常用他们去执行某项特殊的任务:当完成任务之后,他们就会被销毁.这三个数据结构还有一 ...
- C++实现一个简单的双栈队列
双栈队列的原理是用两个栈结构模拟一个队列, 一个栈A模拟队尾, 入队的元素全部压入此栈, 另一个栈B模拟队首, 出队时将栈A的元素弹入栈B, 将栈B的栈顶元素弹出 此结构类似汉诺塔, 非常经典, 这里 ...
- leetcode 栈和队列类型题
1,Valid Parentheses bool isVaild1(string& s) { // 直接列举,不易扩展 stack<char> stk; ; i < s.le ...
- leetcode 栈 括号匹配
https://oj.leetcode.com/problems/valid-parentheses/ 遇到左括号入栈,遇到右括号出栈找匹配,为空或不匹配为空, public class Soluti ...
- 【图解数据结构】 栈&队列
[TOC] 勤于总结,持续输出! 1.栈 1.1栈的定义 栈(stack)是限定在表尾进行插入和删除的操作的线性表. 我们把允许插入和删除的一端称为栈顶(top),另一端称为栈底(bottom),不包 ...
- 数据结构 栈&队列
2-4 依次在初始为空的队列中插入元素a,b,c,d以后,紧接着做了两次删除操作,此时的队头元素是( ) 删除,移动头指针: 增加,移动尾指针: 删除a,b ,队头c 2-3 在一个链队列中,fron ...
随机推荐
- 算法竞赛进阶指南0x14 Hash
组成部分: 哈希函数: 链表 AcWing137. 雪花雪花雪花 因为所需要数据量过于大,所以只能以O(n)的复杂度. 所以不可能在实现的过程中一一顺时针逆时针进行比较,所以采用一种合适的数据结构. ...
- CDH集群日常
1.交换内存告警 该告警通常就是使用了swap分区导致的,在生产环境中,如果服务器内存是256G:建议关闭swap分区,减少跟数据盘之间的交互: 2.修改swap阈值 任何:表示只要使用了swap分区 ...
- typescript中的esModuleInterop选项
当没有加esModuleInterop时 库的代码: export const a = 1; export default function b() {} 生成代码 exports.__esModul ...
- websocket、socket、http对比
简介 在之前的理解中,讲述了socket.websocket等相关的理解,本文就socket.websocket.http理解一下其对应的联系和区别. HTTP 协议 http 为短连接:客户端发送请 ...
- 哎,又跟HR在小群吵了一架!
原创不易,求分享.求一键三连 书接上文: 跟HR在大群吵了一架... 难道,降本增效就是裁员吗 前段时间我问了自己一个问题,如果自己真的是公司内部的外包团队,会怎么样?自从思维转变后,很多事情居然得到 ...
- 关于微信豆苹果(IOS)用户1比10充值方法
微信iOS端微信7.0.20版本之后就上线了微信豆功能,相比大家对微信豆已经不陌生了. 微信官方现在给出了微信豆的含义,微信豆是用于支付微信内虚拟物品的道具,支持在视频号中购买虚拟礼物.也可以对公 ...
- ebook下载 | 《 企业高管IT战略指南——企业为何要落地DevOps》
"当下,企业DevOps转型不仅是IT部门的事情,更是企业高管必须关注的焦点.DevOps是一项需要自上而下推动的变革运动,只有从顶层实施,才能获得成功.本书将介绍企业高管必须了解的,Dev ...
- 一个非常简单用.NET操作RabbitMQ的方法
RabbitMQ作为一款主流的消息队列工具早已广受欢迎.相比于其它的MQ工具,RabbitMQ支持的语言更多.功能更完善. 本文提供一种市面上最/极简单的使用RabbitMQ的方式(支持.NET/.N ...
- 深入分析FragmentPagerAdapter和FragmentStatePagerAdapter
最近遇到比较奇怪的bug,TableLayout+ViewPager实现点击顶部tab切换viewpager视图.但是在Viewpager设置dapter时,最开始设置的是FragmentPagerA ...
- java学习第四天高级数组.day13
正则表达式 冒泡排序 二分法查找 线性查找:从头找到尾,性能比较低. 二分法查找(折半查找):前提数组元素是有序的,性能非常优异. Arrays