华为OD机试 - 空栈压数 - 栈(Java 2024 E卷 100分)

2024-08-24 15:04

本文主要是介绍华为OD机试 - 空栈压数 - 栈(Java 2024 E卷 100分),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

华为OD机试 2024E卷题库疯狂收录中,刷题点这里

专栏导读

本专栏收录于《华为OD机试(JAVA)真题(E卷+D卷+A卷+B卷+C卷)》。

刷的越多,抽中的概率越大,私信哪吒,备注华为OD,加入华为OD刷题交流群,每一题都有详细的答题思路、详细的代码注释、3个测试用例、为什么这道题采用XX算法、XX算法的适用场景,发现新题目,随时更新,全天CSDN在线答疑。

一、题目描述

向一个空栈压入正整数,每当压入一个整数时,执行以下规则(设:栈顶至栈底整数依次为n1, n2, …, nx,其中n1为最新压入的整数)

  1. 如果 n1 = n2,则 n1, n2 全部出栈,压入新数据 m (m = 2 * n1)
  2. 如果 n1 = n2 + … + ny 的范围为 3,n,则 n1, n2, …, ny 全部出栈,压入新数据 m (m = 2 * n1)
  3. 如果上述规则均不满足,则不做操作。

如:依次向栈压入 6,1,2,3,

  1. 当压入 2 时,栈顶至栈底依次为 2,1,6;
  2. 当压入 3 时,3 = 2 + 1,3,2,1 全部出栈,重新入栈整数 6,此时栈顶至栈底依次为 6,6;6 = 6,两个 6 全部出栈,压入 12,最终栈中只剩个元素 12。向栈中输入一串数字,请输出应用此规则后栈中最终存留的数字。

二、输入描述

使用单个空格隔开的正整数的字符串 Q,如“5678”,左边的数字先入栈。

  1. 正整数大小为 [1,231 - 1]
  2. 正整数个数为 [1,1000]。

三、输出描述

最终栈中存留的元素值,元素值使用单个空格隔开,从左至右依次为栈顶至栈底的数字。

四、测试用例

测试用例1:

1、输入

10 20 50 80 1 1

2、输出

2 160

测试用例2:

1、输入

5 10 20 50 85 1

2、输出

1 170

五、解题思路

这道题的核心在于使用栈(Stack)来模拟一个逐步压入数字,并根据特定规则进行元素处理的过程。问题要求我们根据输入的数字列表,按照一定的规则对栈进行操作,最后输出栈中的元素。

1、具体步骤如下:

  1. 输入解析:
    • 从输入字符串中读取一组用空格分隔的正整数。按照顺序将这些整数逐个压入栈中。
  2. 栈操作:
    • 每次压入一个新数字后,检查是否满足以下两个规则:
    • 规则1:如果栈顶两个元素相等,则将这两个元素弹出栈,并将它们的两倍值压入栈中。
    • 规则2:如果栈顶元素等于栈中前几个元素之和(范围为3到n),则将这些元素弹出栈,并将栈顶元素的两倍值压入栈中。
    • 在每次应用规则后,需要重新检查栈是否满足规则(因为栈的状态可能发生了变化),直到不能再应用规则为止。
  3. 结果输出:
    • 当所有输入数字都处理完毕,并且不能再应用任何规则时,将栈中的元素按顺序输出。输出的顺序应从栈顶到栈底。

2、算法复杂度分析

(1)时间复杂度:

由于每个元素可能会多次参与规则的匹配,最坏情况下,时间复杂度可以达到 O(n^2),其中 n 是输入数字的个数。这是因为每次元素的压入可能会触发多个规则匹配和弹出操作,尤其是在规则2中需要遍历栈的部分元素进行求和匹配。

(2)空间复杂度:

空间复杂度主要由栈的使用决定,最坏情况下,需要额外的栈空间来保存所有元素,因此空间复杂度为 O(n)。

六、Java算法源码

public class OdTest01 {public static void main(String[] args) {Scanner sc = new Scanner(System.in);// 读取输入的数字串,并分割成单个数字的数组String input = sc.nextLine();String[] numbers = input.split(" ");// 使用栈来存放数字Stack<Integer> stack = new Stack<>();// 遍历每个数字并压入栈中for (String num : numbers) {int currentNumber = Integer.parseInt(num); // 将字符串转换为整数stack.push(currentNumber); // 压入栈中// 尝试应用规则1和规则2applyRules(stack); // 对栈应用规则}// 输出最终栈中的内容printStack(stack);}// 对栈应用规则1和规则2private static void applyRules(Stack<Integer> stack) {boolean ruleApplied;// 不断检查直到不能再应用规则do {ruleApplied = false;// 规则1:栈顶两个元素相等if (stack.size() >= 2) {int top1 = stack.pop();int top2 = stack.pop();if (top1 == top2) {int newElement = 2 * top1;stack.push(newElement);ruleApplied = true;} else {stack.push(top2); // 恢复栈的状态stack.push(top1);}}// 规则2:栈顶元素等于前若干元素之和if (!ruleApplied && stack.size() >= 3) {int top1 = stack.pop();int sum = 0;Stack<Integer> tempStack = new Stack<>(); // 临时栈用来保存弹出的元素// 计算前面的若干元素之和while (!stack.isEmpty()) {int element = stack.pop();tempStack.push(element); // 保存到临时栈sum += element;if (sum == top1) {int newElement = 2 * top1;stack.push(newElement); // 压入新元素ruleApplied = true;break;}}// 如果没有应用规则2,则恢复栈的状态if (!ruleApplied) {while (!tempStack.isEmpty()) {stack.push(tempStack.pop());}stack.push(top1);}}} while (ruleApplied); // 如果应用了规则,继续检查}// 按照要求输出栈中的元素private static void printStack(Stack<Integer> stack) {// 按栈顶到栈底的顺序输出while (!stack.isEmpty()) {System.out.print(stack.pop());if (!stack.isEmpty()) {System.out.print(" ");}}}
}

七、效果展示

1、输入

10 20 50 80 1 1

2、输出

2 160

3、说明

  1. 压入10:
    • 栈:[10]
    • 没有规则满足,继续。
  2. 压入20:
    • 栈:[10, 20]
    • 没有规则满足,继续。
  3. 压入50:
    • 栈:[10, 20, 50]
    • 没有规则满足,继续。
  4. 压入80:
    • 栈:[10, 20, 50, 80]
    • 没有规则满足,继续。
  5. 压入1:
    • 栈:[10, 20, 50, 80, 1]
    • 没有规则满足,继续。
  6. 再次压入1:
    • 栈:[10, 20, 50, 80, 1, 1]
  7. 触发规则1:栈顶的两个元素相等(两个1),将它们弹出并将它们的两倍值(2)压入栈中。
    • 栈变化为:[10, 20, 50, 80, 2]
  8. 触发规则2:
    • 现在栈顶元素是2,但没有符合规则2的条件(即没有与2相等的前若干元素的和)。
    • 栈保持为:[10, 20, 50, 80, 2]
    • 接下来处理栈中的元素以检查是否有符合规则的。
  9. 再次触发规则2:
    • 栈顶元素是2,无需处理,但栈中有:80、50、20和10,其中,80 = 50 + 20 + 10,满足规则2的条件。
    • 弹出80、50、20、10,然后将160(2*80)压入栈中。
    • 栈最终变为:[160, 2]
  10. 最终栈状态:
    • 经过以上步骤操作,栈中元素从栈顶到栈底依次为:2 和 160。

在这里插入图片描述


🏆下一篇:华为OD机试 - 简易内存池 - 逻辑分析(Java 2024 D卷 200分)

🏆本文收录于,华为OD机试(JAVA)真题(E卷+D卷+A卷+B卷+C卷)

刷的越多,抽中的概率越大,私信哪吒,备注华为OD,加入华为OD刷题交流群,每一题都有详细的答题思路、详细的代码注释、3个测试用例、为什么这道题采用XX算法、XX算法的适用场景,发现新题目,随时更新,全天CSDN在线答疑。

在这里插入图片描述

这篇关于华为OD机试 - 空栈压数 - 栈(Java 2024 E卷 100分)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

JVisualVM之Java性能监控与调优利器详解

《JVisualVM之Java性能监控与调优利器详解》本文将详细介绍JVisualVM的使用方法,并结合实际案例展示如何利用它进行性能调优,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全... 目录1. JVisualVM简介2. JVisualVM的安装与启动2.1 启动JVisualVM2

Java如何从Redis中批量读取数据

《Java如何从Redis中批量读取数据》:本文主要介绍Java如何从Redis中批量读取数据的情况,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一.背景概述二.分析与实现三.发现问题与屡次改进3.1.QPS过高而且波动很大3.2.程序中断,抛异常3.3.内存消

SpringBoot使用ffmpeg实现视频压缩

《SpringBoot使用ffmpeg实现视频压缩》FFmpeg是一个开源的跨平台多媒体处理工具集,用于录制,转换,编辑和流式传输音频和视频,本文将使用ffmpeg实现视频压缩功能,有需要的可以参考... 目录核心功能1.格式转换2.编解码3.音视频处理4.流媒体支持5.滤镜(Filter)安装配置linu

在Spring Boot中实现HTTPS加密通信及常见问题排查

《在SpringBoot中实现HTTPS加密通信及常见问题排查》HTTPS是HTTP的安全版本,通过SSL/TLS协议为通讯提供加密、身份验证和数据完整性保护,下面通过本文给大家介绍在SpringB... 目录一、HTTPS核心原理1.加密流程概述2.加密技术组合二、证书体系详解1、证书类型对比2. 证书获

Java使用MethodHandle来替代反射,提高性能问题

《Java使用MethodHandle来替代反射,提高性能问题》:本文主要介绍Java使用MethodHandle来替代反射,提高性能问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录一、认识MethodHandle1、简介2、使用方式3、与反射的区别二、示例1、基本使用2、(重要)

Java实现本地缓存的常用方案介绍

《Java实现本地缓存的常用方案介绍》本地缓存的代表技术主要有HashMap,GuavaCache,Caffeine和Encahche,这篇文章主要来和大家聊聊java利用这些技术分别实现本地缓存的方... 目录本地缓存实现方式HashMapConcurrentHashMapGuava CacheCaffe

SpringBoot整合Sa-Token实现RBAC权限模型的过程解析

《SpringBoot整合Sa-Token实现RBAC权限模型的过程解析》:本文主要介绍SpringBoot整合Sa-Token实现RBAC权限模型的过程解析,本文给大家介绍的非常详细,对大家的学... 目录前言一、基础概念1.1 RBAC模型核心概念1.2 Sa-Token核心功能1.3 环境准备二、表结

eclipse如何运行springboot项目

《eclipse如何运行springboot项目》:本文主要介绍eclipse如何运行springboot项目问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目js录当在eclipse启动spring boot项目时出现问题解决办法1.通过cmd命令行2.在ecl

Java中的Closeable接口及常见问题

《Java中的Closeable接口及常见问题》Closeable是Java中的一个标记接口,用于表示可以被关闭的对象,它定义了一个标准的方法来释放对象占用的系统资源,下面给大家介绍Java中的Clo... 目录1. Closeable接口概述2. 主要用途3. 实现类4. 使用方法5. 实现自定义Clos

Jvm sandbox mock机制的实践过程

《Jvmsandboxmock机制的实践过程》:本文主要介绍Jvmsandboxmock机制的实践过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、背景二、定义一个损坏的钟1、 Springboot工程中创建一个Clock类2、 添加一个Controller