【Py/Java/C++三种语言OD2023C卷真题】20天拿下华为OD笔试【DP】2023C-分班【欧弟算法】全网注释最详细分类最全的华为OD真题题解

文章目录

  • 题目描述与示例
    • 题目描述
    • 输入描述
    • 输出描述
    • 示例一
      • 输入
      • 输出
    • 示例二
      • 输入
      • 输出
  • 解题思路
  • 代码
    • Python
    • Java
    • C++
    • 时空复杂度
  • 华为OD算法/大厂面试高频题算法练习冲刺训练

题目描述与示例

题目描述

幼儿园两个班的小朋友在排队时混在了一起,每位小朋友都知道自己是否与前面一位小朋友是否同班,请你帮忙把同班的小朋友找出来。小朋友的编号为整数,与前一位小朋友同班用Y表示,不同班用N表示。

输入描述

输入为空格分开的小朋友编号和是否同班标志。

比如: 6/N 2/Y 3/N 4/Y,表示共4位小朋友,26同班, 32不同班,43同班。

其中,小朋友总数不超过999,每个小朋友编号大于0,小于等于999。不考虑输入格式错误问题。

输出描述

输出为两行,每一行记录一个班小朋友的编号,编号用空格分开。

且:

  1. 编号需要按照大小升序排列,分班记录中第一个编号小的排在第一行;
  2. 若只有一个班的小朋友,第二行为空行;
  3. 若输入不符合要求,则直接输出字符串ERROR

示例一

输入

6/N
2/Y
3/N
4/Y

输出

2 6
3 4

示例二

输入

2/N
3/Y
4/Y

输出

2 3 4

解题思路

每一个小朋友在哪个班,只跟其前一个小朋友在哪个班有关系。换句话说,考虑第i个小朋友的分班,我们只需要关心第i-1个小朋友在哪个班,而不需要关心第i-1个小朋友为什么会分配到这个班。这是一种典型的dp思路。

由于一共只有两个班,为了方便起见,我们分为1班和2班,同时默认第0个小朋友位于1班。

我们考虑动态规划三部曲:

  1. dp数组的含义是什么?
  • dp数组是一个长度为n的一维列表,dp[i]是布尔变量True或者False。如果
    • dp[i] = True表示第i个小朋友分在1班
    • dp[i] = False表示第i个小朋友分在2班
  1. 动态转移方程是什么?
  • dp[i]只跟dp[i-1]以及YN_lst[i]有关系。若
    • i个小朋友和第i-1个小朋友同班,即YN_lst[i] == "Y",那么dp[i] = dp[i-1]
    • i个小朋友和第i-1个小朋友不同班,即YN_lst[i] == "N",那么dp[i] = not dp[i-1]
if YN_lst[i] == "Y":dp[i] = dp[i-1]
elif YN_lst[i] == "N":dp[i] = not dp[i-1]
  1. dp数组如何初始化?
  • 默认第0个小朋友是在1班,故dp[0] = True

做完dp过程之后,还需要依据题目要求,对两个班的小朋友进行排序和输出。其过程如下

class1, class2 = list(), list()
for i in range(n):class1.append(children_lst[i]) if dp[i] else class2.append(children_lst[i])class1.sort()
class2.sort()
if len(class2) == 0:print(" ".join(map(str, class1)))print("")
else:if class1[0] > class2[0]:class1, class2 = class2, class1print(" ".join(map(str, class1)))print(" ".join(map(str, class2)))

代码

Python

# 题目:2023B-分班
# 分值:100
# 作者:许老师-闭着眼睛学数理化
# 算法:DP
# 代码看不懂的地方,请直接在群上提问children_lst = list()
YN_lst = list()# 由于不知道输入行数,故使用try-except语句,结合while死循环进行输入
while True:# try语句,尝试执行以下语句try:# 进行输入s = input()# 如果s无法进行分割成两部分,则此处解包会出现报错,进入except语句child, ch = s.split("/")# 成功分割,则将小朋友编号记录在children_lst中# 是否与前一个小朋友同班的标记"Y"或"N"记录在YN_lst中children_lst.append(int(child))YN_lst.append(ch)# 如果出现错误,说明输入最后一行,退出死循环except:break# 获得小朋友的数目
n = len(children_lst)# 构建长度为n的dp数组,dp[i]的取值为True和False
# True表示第i个小朋友在1班,False表示第i个小朋友在2班
dp = [None] * n
# 初始化dp[0]为True,表示默认分配第0个小朋友在1班
dp[0] = True# 遍历所有小朋友
for i in range(1, n):# 如果YN_lst[i]为"Y",表示第i个小朋友和第i-1个小朋友同班# dp[i]和dp[i-1]取相同if YN_lst[i] == "Y":dp[i] = dp[i-1]# 如果YN_lst[i]为"N",表示第i个小朋友和第i-1个小朋友不同班# dp[i]和dp[i-1]取相反elif YN_lst[i] == "N":dp[i] = not dp[i-1]# 构建表示两个班级的列表
class1, class2 = list(), list()
for i in range(n):# 如果dp[i]为True,分到class1,如果dp[i]为False,分到class2class1.append(children_lst[i]) if dp[i] else class2.append(children_lst[i])# 题目要求排序,故对class1和class2进行排序
class1.sort()
class2.sort()
# 如果class2长度为0,说明所有小朋友都在1班
# 输出1班的结果和一个空字符串
if len(class2) == 0:print(" ".join(map(str, class1)))print("")
else:# 两个班级之间的排序,以班级中编号最小的小朋友为准# 故如果发现2班编号最小的小朋友比1班编号最小的小朋友更小# 应该先输出2班,故对两者进行交换if class1[0] > class2[0]:class1, class2 = class2, class1# 先输出1班,再输出2班print(" ".join(map(str, class1)))print(" ".join(map(str, class2)))

Java

import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;public class Main {public static void main(String[] args) {List<Integer> childrenList = new ArrayList<>();List<String> ynList = new ArrayList<>();// 由于不知道输入行数,故使用 try-catch 语句,结合 while 死循环进行输入Scanner scanner = new Scanner(System.in);while (true) {// try 语句,尝试执行以下语句try {// 进行输入String s = scanner.nextLine();// 如果 s 无法进行分割成两部分,则此处解包会出现报错,进入 catch 语句String[] parts = s.split("/");int child = Integer.parseInt(parts[0]);String ch = parts[1];// 成功分割,则将小朋友编号记录在 childrenList 中// 是否与前一个小朋友同班的标记 "Y" 或 "N" 记录在 ynList 中childrenList.add(child);ynList.add(ch);}// 如果出现错误,说明输入最后一行,退出死循环catch (Exception e) {break;}}// 获得小朋友的数目int n = childrenList.size();// 构建长度为 n 的 dp 数组,dp[i] 的取值为 true 和 false// true 表示第 i 个小朋友在 1 班,false 表示第 i 个小朋友在 2 班boolean[] dp = new boolean[n];// 初始化 dp[0] 为 true,表示默认分配第 0 个小朋友在 1 班dp[0] = true;// 遍历所有小朋友for (int i = 1; i < n; i++) {// 如果 ynList[i] 为 "Y",表示第 i 个小朋友和第 i-1 个小朋友同班// dp[i] 和 dp[i-1] 取相同if (ynList.get(i).equals("Y")) {dp[i] = dp[i - 1];}// 如果 ynList[i] 为 "N",表示第 i 个小朋友和第 i-1 个小朋友不同班// dp[i] 和 dp[i-1] 取相反else if (ynList.get(i).equals("N")) {dp[i] = !dp[i - 1];}}// 构建表示两个班级的列表List<Integer> class1 = new ArrayList<>();List<Integer> class2 = new ArrayList<>();for (int i = 0; i < n; i++) {// 如果 dp[i] 为 true,分到 class1,如果 dp[i] 为 false,分到 class2if (dp[i]) {class1.add(childrenList.get(i));} else {class2.add(childrenList.get(i));}}// 题目要求排序,故对 class1 和 class2 进行排序class1.sort(Integer::compareTo);class2.sort(Integer::compareTo);// 如果 class2 长度为 0,说明所有小朋友都在 1 班// 输出 1 班的结果和一个空字符串if (class2.isEmpty()) {for (int child : class1) {System.out.print(child + " ");}System.out.println();System.out.println();} else {// 两个班级之间的排序,以班级中编号最小的小朋友为准// 故如果发现 2 班编号最小的小朋友比 1 班编号最小的小朋友更小// 应该先输出 2 班,故对两者进行交换if (class1.get(0) > class2.get(0)) {List<Integer> temp = class1;class1 = class2;class2 = temp;}// 先输出 1 班,再输出 2 班for (int child : class1) {System.out.print(child + " ");}System.out.println();for (int child : class2) {System.out.print(child + " ");}System.out.println();}}
}

C++

#include <iostream>
#include <vector>
#include <algorithm>using namespace std;int main() {vector<int> childrenList;vector<string> ynList;// 由于不知道输入行数,故使用 try-catch 语句,结合 while 死循环进行输入while (true) {// try 语句,尝试执行以下语句try {// 进行输入string s;getline(cin, s);// 如果 s 无法进行分割成两部分,则此处解包会出现报错,进入 catch 语句int pos = s.find('/');int child = stoi(s.substr(0, pos));string ch = s.substr(pos + 1);// 成功分割,则将小朋友编号记录在 childrenList 中// 是否与前一个小朋友同班的标记 "Y" 或 "N" 记录在 ynList 中childrenList.push_back(child);ynList.push_back(ch);}// 如果出现错误,说明输入最后一行,退出死循环catch (exception& e) {break;}}// 获得小朋友的数目int n = childrenList.size();// 构建长度为 n 的 dp 数组,dp[i] 的取值为 true 和 false// true 表示第 i 个小朋友在 1 班,false 表示第 i 个小朋友在 2 班vector<bool> dp(n);// 初始化 dp[0] 为 true,表示默认分配第 0 个小朋友在 1 班dp[0] = true;// 遍历所有小朋友for (int i = 1; i < n; i++) {// 如果 ynList[i] 为 "Y",表示第 i 个小朋友和第 i-1 个小朋友同班// dp[i] 和 dp[i-1] 取相同if (ynList[i] == "Y") {dp[i] = dp[i - 1];}// 如果 ynList[i] 为 "N",表示第 i 个小朋友和第 i-1 个小朋友不同班// dp[i] 和 dp[i-1] 取相反else if (ynList[i] == "N") {dp[i] = !dp[i - 1];}}// 构建表示两个班级的列表vector<int> class1, class2;for (int i = 0; i < n; i++) {// 如果 dp[i] 为 true,分到 class1,如果 dp[i] 为 false,分到 class2if (dp[i]) {class1.push_back(childrenList[i]);} else {class2.push_back(childrenList[i]);}}// 题目要求排序,故对 class1 和 class2 进行排序sort(class1.begin(), class1.end());sort(class2.begin(), class2.end());// 如果 class2 长度为 0,说明所有小朋友都在 1 班// 输出 1 班的结果和一个空行if (class2.empty()) {for (int child : class1) {cout << child << " ";}cout << endl << endl;} else {// 两个班级之间的排序,以班级中编号最小的小朋友为准// 故如果发现 2 班编号最小的小朋友比 1 班编号最小的小朋友更小// 应该先输出 2 班,故对两者进行交换if (class1[0] > class2[0]) {swap(class1, class2);}// 先输出 1 班,再输出 2 班for (int child : class1) {cout << child << " ";}cout << endl;for (int child : class2) {cout << child << " ";}cout << endl;}return 0;
}

时空复杂度

时间复杂度:O(xlogx + ylogy + N)O(xlogx)O(ylogy)是两个班级各自排序的时间复杂度,O(N)是dp过程的时间复杂度。

空间复杂度:O(N)

xy为两个班级的人数,N是总人数。


华为OD算法/大厂面试高频题算法练习冲刺训练

  • 华为OD算法/大厂面试高频题算法冲刺训练目前开始常态化报名!目前已服务100+同学成功上岸!

  • 课程讲师为全网50w+粉丝编程博主@吴师兄学算法 以及小红书头部编程博主@闭着眼睛学数理化

  • 每期人数维持在20人内,保证能够最大限度地满足到每一个同学的需求,达到和1v1同样的学习效果!

  • 60+天陪伴式学习,40+直播课时,300+动画图解视频,300+LeetCode经典题,200+华为OD真题/大厂真题,还有简历修改、模拟面试、专属HR对接将为你解锁

  • 可上全网独家的欧弟OJ系统练习华子OD、大厂真题

  • 可查看链接 大厂真题汇总 & OD真题汇总(持续更新)

  • 绿色聊天软件戳 od1336了解更多

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

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

相关文章

【十】python复合模式

10.1 复合模式简介 在前面的栏目中我们了解了各种设计模式。正如我们所看到的&#xff0c;设计模式可分为三大类:结构型、创建型和行为型设计模式。同时&#xff0c;我们还给出了每种类型的相应示例。然而&#xff0c;在软件实现中&#xff0c;模式并是不孤立地工作的。对于所…

HPM5300系列--第一篇 命令行开发调试环境搭建

一、目的 在之前的博客中《HPM6750系列--第二篇 搭建Ubuntu开发环境》、 《HPM6750系列--第三篇 搭建MACOS编译和调试环境》我们介绍了HPM6750evkmini开发环境的搭建过程&#xff0c;由于HPM5300系列共用一套hpm-sdk&#xff0c;故HPM5300的开发调试环境的搭建过程基本和之前的…

golang学习笔记——互斥锁sync.Mutex、计数器sync.WaitGroup、读写锁sync.RWMutex

文章目录 互斥锁&#xff1a; sync.Mutexsync.WaitGroup 计数器例子func (*WaitGroup) Addfunc (*WaitGroup) Donefunc (*WaitGroup) Wait 读写互斥锁参考资料 临界区总是需要通过同步机制进行保护的&#xff0c;否则就会产生竞态条件&#xff0c;导致数据不一致。 互斥锁&…

智能故障诊断期刊推荐【中文期刊】

控制与决策 http://kzyjc.alljournals.cn/kzyjc/home 兵工学报 http://www.co-journal.com/CN/1000-1093/home.shtml 计算机集成制造系统 http://jsjjc.soripan.net/ 机械工程学报 http://www.cjmenet.com.cn/CN/0577-6686/home.shtml 太阳能学报 https://www.tynxb.org.c…

Visual Studio Code中的任务配置文件tasks.json中的可选任务组tasks详解

☞ ░ 前往老猿Python博客 ░ https://blog.csdn.net/LaoYuanPython 一、引言 vscode是支持通过配置可以实现类似Visual C等IDE开发工具使用菜单和快捷键直接进行程序编译构建的&#xff0c;这样构建的任务可以结合后续的调试配置进行IDE环境的程序调试&#xff0c;不过在之前…

12. IO

1.File类 • File 类代表与平台无关的文件和目录。 • File 能新建、删除、重命名文件和目录&#xff0c;但 File 不能访问文件内容本身。如果需要访问文件内容本身&#xff0c;则需要使用输入/输出流。 1).File的常用方法 在这里插入图片描述 2).遍历给定目录所有文件 …

Android源码下载流程

1.使用repo方式&#xff1a; https://github.com/jeanboydev/Android-ReadTheFuckingSourceCode/blob/master/article/android/framework/Android-Windows%E7%8E%AF%E5%A2%83%E4%B8%8B%E8%BD%BD%E6%BA%90%E7%A0%81.md 2.使用git方式&#xff1a; Windows 环境下载 Android 源…

QT之QMatrix

QT之QMatrix 成员函数介绍使用注意函数的使用 成员函数介绍 reset()&#xff1a;将矩阵重置为单位矩阵&#xff08;无变换&#xff09;。 translate()&#xff1a;执行平移变换。例如&#xff0c;translate(qreal dx, qreal dy)将图形沿x轴和y轴方向分别平移dx和dy个单位。 sc…

MySQL增删改查

查询数据 MySQL 数据库使用 SQL SELECT 语句来查询数据。以下为在 MySQL 数据库中查询数据通用的 SELECT 语法&#xff1a; SELECT column_name,column_name FROM table_name [WHERE Clause] [LIMIT N][ OFFSET M] 查询语句中你可以使用一个或者多个表&#xff0c;表之间使用…

联想笔记本如何安装Vmware ESXi

环境&#xff1a; Vmware ESXi 8.0 Vmware ESXi 6.7 联想E14笔记本 问题描述&#xff1a; 联想笔记本如何安装Vmware ESXi 解决方案&#xff1a; 1.官网下载镜像文件 https://customerconnect.vmware.com/en/downloads/search?queryesxi%208 下载 2.没有账户注册一个 …

什么时候使用匿名类,匿名类解决了什么问题?为什么需要匿名类 ?

匿名类通常在以下场景下使用&#xff1a; 一次性使用&#xff1a; 当你需要创建一个类的实例&#xff0c;但该类只在一个地方使用&#xff0c;而不打算在其他地方重复使用时&#xff0c;可以考虑使用匿名类。 简化代码&#xff1a; 当创建一个小型的、一次性的类会让代码更简洁…

浅析特征增强个性化在CTR预估中的经典方法和效果对比

在CTR预估中&#xff0c;主流都采用特征embeddingMLP的方式&#xff0c;其中特征非常关键。然而对于相同的特征&#xff0c;在不同的样本中&#xff0c;表征是相同的&#xff0c;这种方式输入到下游模型&#xff0c;会限制模型的表达能力。为了解决这个问题&#xff0c;CTR预估…

12.14 log 376. 摆动序列,53. 最大子序和

376. 摆动序列 class Solution { public:int wiggleMaxLength(vector<int>& nums) {if(nums.size()1) return nums.size();int preDiff0;int curDiff0;int result1;for(int i0;i<nums.size()-1;i){curDiffnums[i1]-nums[i];if(preDiff<0&&curDiff>…

【每日一题】用邮票贴满网格图

文章目录 Tag题目来源题目解读解题思路方法一&#xff1a;二维前缀和二维差分 写在最后 Tag 【二维前缀和】【二维差分】【矩阵】【2023-12-14】 题目来源 2132. 用邮票贴满网格图 题目解读 在 01 矩阵中&#xff0c;判断是否可以用给定尺寸的邮票将所有 0 位置都覆盖住&…

智能优化算法应用:基于黄金正弦算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于黄金正弦算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于黄金正弦算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.黄金正弦算法4.实验参数设定5.算法结果6.…

GeoTrust OV证书

当谈到网站安全性和可信度时&#xff0c;GeoTrust OV证书是一个备受推崇的选择。作为一家备受尊敬的数字证书颁发机构&#xff0c;GeoTrust以其卓越的品牌声誉和高质量的产品而闻名于世。GeoTrust OV证书提供了一系列的安全功能&#xff0c;同时还具有出色的性价比&#xff0c;…

再探再报:SQL中的关联查询

在 SQL 中进行关联查询&#xff08;或称联接查询&#xff09;时&#xff0c;有几种不同的语法来执行这些查询&#xff0c;常见的包括&#xff1a; 使用 JOIN 关键字&#xff1a;JOIN 关键字可与不同类型的联接组合使用&#xff0c;如 INNER JOIN、LEFT JOIN、RIGHT JOIN、FULL …

DPDK多进程之间的通信

文章目录 前言本机DPDK IPC API介绍demo演示 前言 DPDK的主进程和辅助进程之间共享大页内存。关于DPDK多进程的支持文档介绍见&#xff1a;47. 多进程支持。 本文介绍本机DPDK的主进程和辅助进程之间交换短消息的API的使用。 前置要求&#xff1a;DPDK-Hello-World示例应用程…

系统级基础信号知识【Linux】

目录 一&#xff0c;什么是信号 进程面对信号常见的三种反应概述 二&#xff0c;产生信号 1.终端按键产生信号 signal 2. 进程异常产生信号 核心转储 3. 系统调用函数发送信号 kill raise abort 小结&#xff1a; 4. 由软件条件产生 alarm 5. 硬件异常产生信号…

解锁MSSQL存储过程优化之道:参数化查询的技术深度探究

数据库是现代应用的支柱&#xff0c;而MSSQL作为其中的瑞士军刀&#xff0c;其性能直接关系到系统的稳定与响应速度。本文将带领读者深入探讨MSSQL存储过程优化的精髓之一——参数化查询。跟随着我们的脚步&#xff0c;你将领悟到优化的本质&#xff0c;发现隐藏在参数化查询背…