LeetCode——栈和队列(Java)

栈和队列

  • 简介
  • [简单] 232. 用栈实现队列
  • [简单] 225. 用队列实现栈
  • [简单] 20. 有效的括号
  • [简单] 1047. 删除字符串中的所有相邻重复项
  • [中等] 150. 逆波兰表达式求值
  • [困难] 239. 滑动窗口最大值
  • [中等] 347. 前 K 个高频元素

简介

记录一下自己刷题的历程以及代码。写题过程中参考了 代码随想录。会附上一些个人的思路,如果有错误,可以在评论区提醒一下。

涉及到:栈、队列、双端队列、map、优先队列、java装箱拆箱问题。

[简单] 232. 用栈实现队列

原题链接

使用两个栈实现,stack1接受数据,每次要pop或者peek的时候,把stack1里的数据再出栈入栈 放置到stack2中,这样又对stack1中的数据做了一次转置。注意stack2中有元素时,stack1不能向stack2传入数据,否则会出现顺序错乱。

class MyQueue {//假设了所有操作都是有效的public Stack<Integer> stack1;public Stack<Integer> stack2;public MyQueue() {stack1 = new Stack<>();stack2 = new Stack<>();}public void push(int x) {stack1.push(x);}public int pop() {reverse();return stack2.pop();}public int peek() {reverse();return stack2.peek();}public boolean empty() {if(stack2.empty() && stack1.empty()){return true;}return false;}//stack2中没有元素时,把stack1中元素全部转置private void reverse(){if(stack2.empty()) {while (!stack1.empty()) {Integer temp = stack1.pop();stack2.push(temp);}}}
}

[简单] 225. 用队列实现栈

原题链接

使用两个队列实现,时刻保持只有一个队列A存放元素,另一个队列B用于pop或者top操作时接收A的数据,就可以对队列A队尾的元素进行操作。队列A和队列B是相对概念,两个队列某一时刻都可能是存放元素的那个队列。

class MyStack {Queue<Integer> queue1;Queue<Integer> queue2;public MyStack() {queue1 = new LinkedList<Integer>();queue2 = new LinkedList<Integer>();}public void push(int x) {if(queue1.isEmpty()) queue2.add(x);else queue1.add(x);}public int pop() {if(queue1.isEmpty()) {while(queue2.size() > 1){queue1.add(queue2.poll());}return queue2.poll();}while(queue1.size() > 1){queue2.add(queue1.poll());}return queue1.poll();}public int top() {if(queue1.isEmpty()) {while(queue2.size() > 1){queue1.add(queue2.poll());}Integer temp =  queue2.peek();queue1.add(queue2.poll());return temp;}while(queue1.size() > 1){queue2.add(queue1.poll());}Integer temp =  queue1.peek();queue2.add(queue1.poll());return temp;}public boolean empty() {if(queue1.isEmpty() && queue2.isEmpty()){return true;}return false;}
}

[简单] 20. 有效的括号

原题链接

使用栈做符号匹配,碰到左括号压栈,碰到右括号,看目前栈顶是否与之匹配,若不匹配或空,则表示匹配错误。若循环结束栈不为空,表示有多余的左括号。

class Solution {public boolean isValid(String s) {Stack<Character> stack = new Stack<>();for(int i = 0; i < s.length(); i++){if(s.charAt(i) == '[' || s.charAt(i) == '{' || s.charAt(i) == '('){stack.push(s.charAt(i));}else if(s.charAt(i) == ']'){if(stack.empty() || stack.peek() != '[') return false;stack.pop();}else if(s.charAt(i) == '}'){if(stack.empty() || stack.peek() != '{') return false;stack.pop();}else if(s.charAt(i) == ')'){if(stack.empty() || stack.peek() != '(') return false;stack.pop();}}if(stack.empty()) return true;return false;}
}

[简单] 1047. 删除字符串中的所有相邻重复项

原题链接

方法①:使用队列,每次入栈时如果与栈顶元素相同,则不入栈,并弹出栈顶元素。

class Solution {public String removeDuplicates(String s) {ArrayDeque<Character> stack = new ArrayDeque<>();for(int i = 0; i < s.length(); i++){if(!stack.isEmpty() && stack.peek() == s.charAt(i)){stack.pop();}else{stack.push(s.charAt(i));}}String str = "";//剩余的元素即为不重复的元素while (!stack.isEmpty()) {str = stack.pop() + str;}return str;}
}

方法②:快慢指针法指针法,就把[0-slow]范围内看作一个栈,栈顶是slow指针的位置,一旦出现c[slow] == c[slow - 1],就回退一次,将栈顶弹出,否则就将当前fast遍历到的元素加进来。

class Solution {//快慢指针法public String removeDuplicates(String s) {//声明字符串数组用于修改,String类是静态变量char[] c = s.toCharArray();int slow = 0;int fast = 0;while(fast < c.length){c[slow] = c[fast];if(slow > 0 && c[slow] == c[slow - 1]) {slow--;} else {slow++;}fast++;}return new String(c, 0, slow);}
}

[中等] 150. 逆波兰表达式求值

原题链接

token是数字,就做类型转换压入栈,如果是运算符,弹出栈顶两个数字做运算,记住减法和除法是有顺序的。

class Solution {public int evalRPN(String[] tokens) {ArrayDeque<Integer> stack = new ArrayDeque<>();for(int i = 0; i < tokens.length; i++){if(tokens[i].equals("+")){int temp1 = stack.pop();int temp2 = stack.pop();stack.push(temp1 + temp2);}else if(tokens[i].equals("-")){int temp1 = stack.pop();int temp2 = stack.pop();stack.push(temp2 - temp1);}else if(tokens[i].equals("*")){int temp1 = stack.pop();int temp2 = stack.pop();stack.push(temp1 * temp2);}else if(tokens[i].equals("/")){int temp1 = stack.pop();int temp2 = stack.pop();stack.push(temp2 / temp1);}else{stack.push(Integer.parseInt(tokens[i]));}}return stack.pop();}
}

[困难] 239. 滑动窗口最大值

原题链接

原先pop方法和push方法都是Integer参数,一半示例无法通过,改成int之后顺利通过。

涉及到装箱拆箱问题,==运算符可以应用与对象包装器对象,只不过检测的事对象是否指向同一个存储区域。比如Integer a = 1000;Integer b =1000a==b 未必是true。

缓存机制:Java 基本数据类型的包装类型的大部分都用到了缓存机制来提升性能。Byte,Short,Integer,Long 这 4 种包装类默认创建了数值 [-128,127] 的相应类型的缓存数据
在这里插入图片描述

class MoQueue{public ArrayDeque<Integer> queue;public MoQueue(){queue = new ArrayDeque<Integer>();}public void pop(int a){if(!queue.isEmpty() && queue.peek() == a)queue.pollFirst();return;}public void push(int a){while(!queue.isEmpty() && a > queue.getLast()){queue.pollLast();}queue.addLast(a);}public Integer peek(){return queue.peek();}
}
class Solution {public int[] maxSlidingWindow(int[] nums, int k) {MoQueue moQueue = new MoQueue();int[] answer = new int[nums.length - k + 1];for(int i = 0; i < k; i++){moQueue .push(nums[i]);}//放入第一个窗口最大值answer[0] = moQueue.peek();for(int i = k; i < nums.length; i++){moQueue.pop(nums[i - k]);moQueue.push(nums[i]);answer[i - k + 1] = moQueue.peek();}return answer;}
}

[中等] 347. 前 K 个高频元素

原题链接

使用map存储键值对记录每个数字出现的频次,再使用优先队列定义排序,获取前k个队头元素即可。

class Solution {public int[] topKFrequent(int[] nums, int k) {Map<Integer,Integer> map = new HashMap<>();//key为数组元素值,val为对应出现次数for(int num:nums){map.put(num,map.getOrDefault(num,0)+1);}PriorityQueue<Integer> pq = new PriorityQueue<>(new Comparator<Integer>() {@Overridepublic int compare(Integer a, Integer b) {return map.get(b) - map.get(a);}});int[] ans = new int[k];int i = 0;for(Integer key : map.keySet()){pq.add(key);}while(i < k){ans[i++] = pq.remove();}return ans;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/499634.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

通过多进程并发方式(fork)实现服务器

以下内容为视频学习记录。 1、父进程accept后返回的文件描述符为cfd以及用于创建连接的lfd; 调用fork()创建子进程后&#xff0c;子进程继承cfd,lfd&#xff0c;通过该cfd与连接过来的客户端通信,lfd对子进程来说没用&#xff0c;可以直接close(lfd); 对于父进程来说&#x…

【MySQL | 第一篇】undo log、redo log、bin log三者之间的区分?

undo log、redo log、bin log三者之间的区分&#xff1f; 从 产生的时间点、日志内容、用途 三方面展开论述即可 1.undo log——撤销日志 时间点&#xff1a;事务开始之前产生&#xff0c;根据当前版本的数据生成一个undo log&#xff0c;也保存在事务开始之前 作用&#xf…

LeetCode:2867. 统计树中的合法路径数目(筛质数+ DFS Java)

目录 2867. 统计树中的合法路径数目 题目描述&#xff1a; 实现代码与思路&#xff1a; 筛质数 DFS 原理思路&#xff1a; 2867. 统计树中的合法路径数目 题目描述&#xff1a; 给你一棵 n 个节点的无向树&#xff0c;节点编号为 1 到 n 。给你一个整数 n 和一个长度为 …

Gophish+EwoMail 自建钓鱼服务器

GophishEwoMail 自建钓鱼服务器 文章目录 GophishEwoMail 自建钓鱼服务器1.前提准备2.搭建EwoMail邮件服务器1&#xff09;Centos7 防火墙操作2&#xff09;设置主机名3&#xff09;host配置4&#xff09;安装EwoMail5&#xff09;获取DKIM6&#xff09;端口服务介绍7&#xff…

1.2 debug的六种指令的使用,四个通用寄存器

汇编语言 首先进入环境 mount c d:masm //把c挂载在d盘中的masm当中 c: //进入c&#xff0c;进入到编译环境 dir //查看文件&#xff0c;可有可无Debug是DOS、Windows都提供的实模式&#xff08;8086 方式&#xff09;程序的调试工具。使用它可以查看CPU各种寄存器中的内容…

如何提取测试点

如何提取测试点 首先会想到从需求文档中提取测试点&#xff0c;每一次迭代之后&#xff0c;都会有需求&#xff0c;需求经理评审之后&#xff0c;我们要基于需求去写测试计划&#xff0c;包括梳理出来的测试点&#xff0c;梳理完测试点之后&#xff0c;编写对应的测试用例&…

Linux添加用户分组练习

一、复制/etc/skel目录为/home/tuser1&#xff08;/home/tuser1及其内部文件的属组和其它用户均没有任何访问权限&#xff09;。 cp -a /etc/skel /home/tuser1 chown -R tuser1:tuser1 /home/tuser1 chmod -R 700 /home/tuser1 二、编辑/etc/group文件&#xff0c;添加组h…

租赁小程序|租赁系统|租赁软件开发带来高效运营

随着社会的不断发展和科技的不断进步&#xff0c;越来越多的企业开始关注设备租赁业务。设备租赁作为一种短期使用设备的方式&#xff0c;为企业提供了灵活和成本节约的优势。针对设备租赁业务的管理和提升企业竞争力的需求&#xff0c;很多企业选择定制开发设备租赁系统。本文…

抖音作品评论id提取工具|视频内容提取软件

抖音视频提取便捷高效&#xff0c;抖音作品评论id提取工具助您快速获取数据 针对抖音作品评论id提取的需求&#xff0c;我们推出了一款功能强大的工具&#xff0c;旨在帮助用户快速提取抖音作品的评论id。无论您是进行数据分析、社交媒体研究还是其他用途&#xff0c;我们的工…

Linux------进程地址空间

目录 一、进程地址空间 二、地址空间本质 三、什么是区域划分 四、为什么要有地址空间 1.让进程以统一的视角看到内存 2.进程访问内存的安全检查 3.将进程管理与内存管理进行解耦 一、进程地址空间 在我们学习C/C的时候&#xff0c;一定经常听到数据存放在堆区、栈区、…

Linux中如何在创建子线程的时候设置为分离属性

#include<stdio.h> #include<stdlib.h> #include<string.h> #include<sys/types.h> #include<unistd.h> #include <pthread.h> void *mythread(void *arg) {printf("id[%ld]\n",pthread_self()); } int main() { //定义pthread_…

【InternLM 实战营笔记】基于 InternLM 和 LangChain 搭建你的知识库

准备环境 bash /root/share/install_conda_env_internlm_base.sh InternLM升级PIP # 升级pip python -m pip install --upgrade pippip install modelscope1.9.5 pip install transformers4.35.2 pip install streamlit1.24.0 pip install sentencepiece0.1.99 pip install a…

定时任务框架

定时任务的框架有哪些 ● Timer&#xff0c;JDK自带的&#xff0c;比较简单&#xff0c;使用的时候&#xff0c;定义一个TimerTask&#xff0c;实现run方法&#xff0c;然后定义一个Timer类&#xff0c;调用timer.schedule(timerTask,1000,3000); ○ 缺点&#xff1a;单线程、…

发工资(个人学习笔记黑马学习)

某公司&#xff0c;账户余额有1W元&#xff0c;给20名员工发工资 员工编号从1到20&#xff0c;从编号1开始&#xff0c;依次领取工资&#xff0c;每人可领取1000元领工资时&#xff0c;财务判断员工的绩效分(1-10)(随机生成)&#xff0c;如果低于5&#xff0c;不发工资&#xf…

腾讯云优惠券领取的三个渠道,先领券再下单!

腾讯云代金券领取渠道有哪些&#xff1f;腾讯云官网可以领取、官方媒体账号可以领取代金券、完成任务可以领取代金券&#xff0c;大家也可以在腾讯云百科蹲守代金券&#xff0c;因为腾讯云代金券领取渠道比较分散&#xff0c;腾讯云百科txybk.com专注汇总优惠代金券领取页面&am…

计算机设计大赛 深度学习卷积神经网络垃圾分类系统 - 深度学习 神经网络 图像识别 垃圾分类 算法 小程序

文章目录 0 简介1 背景意义2 数据集3 数据探索4 数据增广(数据集补充)5 垃圾图像分类5.1 迁移学习5.1.1 什么是迁移学习&#xff1f;5.1.2 为什么要迁移学习&#xff1f; 5.2 模型选择5.3 训练环境5.3.1 硬件配置5.3.2 软件配置 5.4 训练过程5.5 模型分类效果(PC端) 6 构建垃圾…

Python算法100例-2.8 猜牌术

完整源代码项目地址&#xff0c;关注博主私信源代码后可获取 1.问题描述2.问题分析3.算法设计4.确定程序框架5.完整的程序6.运行结果 1&#xff0e;问题描述 魔术师利用一副牌中的13张黑桃&#xff0c;预先将它们排好后叠在一起&#xff0c;并使牌面朝下。然后他对观众说&am…

04 Opencv图像操作

文章目录 读写像素修改像素值Vec3b与Vec3F灰度图像增强获取图像通道bitwise_not 算子对图像非操作 读写像素 读一个GRAY像素点的像素值&#xff08;CV_8UC1&#xff09; Scalar intensity img.at(y, x); 或者 Scalar intensity img.at(Point(x, y)); 读一个RGB像素点的像素值…

Linux-Uboot命令

help命令 进入 uboot 的命令行模式后输入“help”或者“&#xff1f;”&#xff0c;然后按下回车即可查看当前 uboot 所支持的命令。 查看某一个命令的帮助信息&#xff1a;&#xff1f;命令名称 或 help命令名称 信息查询命令 常用的和信息查询有关的命令有 3 个…

代码随想录算法训练营第三十二天|122.买卖股票的最佳时机II、55. 跳跃游戏、45.跳跃游戏II

122.买卖股票的最佳时机II 刷题https://leetcode.cn/problems/best-time-to-buy-and-sell-stock-ii/description/文章讲解https://programmercarl.com/0122.%E4%B9%B0%E5%8D%96%E8%82%A1%E7%A5%A8%E7%9A%84%E6%9C%80%E4%BD%B3%E6%97%B6%E6%9C%BAII.html视频讲解https://www.bi…