华为OD机试 - 区间交集 - 深度优先搜索dfs算法(滥用)(Java 2023 B卷 200分)

news2024/10/5 23:29:28

在这里插入图片描述

目录

    • 专栏导读
    • 一、题目描述
    • 二、输入描述
    • 三、输出描述
      • 备注
      • 用例
        • 1、输入
        • 2、输出
        • 3、说明
    • 四、解题思路
      • 1、核心思路:
      • 2、具体步骤
    • 五、Java算法源码
      • 再重新读一遍题目,看看能否优化一下~
      • 解题步骤也简化了很多。
    • 六、效果展示
      • 1、输入
      • 2、输出
      • 3、说明

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

专栏导读

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

刷的越多,抽中的概率越大,每一题都有详细的答题思路、详细的代码注释、样例测试,发现新题目,随时更新,全天CSDN在线答疑。

一、题目描述

给定一组闭区间,其中部分区间存在交集。

任意两个给定区间的交集,称为公共区间(如:[1,2],[2,3]的公共区间为[2,2],[3,5],[3,6]的公共区间为[3,5])公共区间之间若存在交集,则需要合并(如:[1,3],[3,5]区间存在交集[3,3],需合并为[1,5])。按升序排列输出合并后的区间列表。

二、输入描述

组区间列表

区间数为 N: O<=N<=1000。

区间元素为 X:-10000<=X<=10000。

三、输出描述

升序排列的合并区间列表

备注

  1. 区间元素均为数字,不考虑字母、符号等异常输入。
  2. 单个区间认定为无公共区间。

用例

1、输入

4
0 3
1 3
3 5
3 6

2、输出

[1,5]

3、说明
  • 0 3和1 3的公共区间是1 3
  • 0 3和3 5的公共区间是3 3
  • 0 3和3 6的公共区间是3 3
  • 1 3和3 5的公共区间是3 3
  • 1 3和3 6的公共区间是3 3
  • 3 5和3 6的公共区间是3 5
  • 两两组合求出公共区间,去重,变为[1,3][3,3][3,5]
  • 若公共区间存在交集,合并为[1,5]

四、解题思路

第一反应是通过深度优先搜索dfs算法来解。

1、核心思路:

  1. 两两组合求出公共区间,左右分别相比,谁小取谁,删除重复的公共区间 [1,3][3,3][3,5]
  2. 有交集就合并,没交集就直接输出,左边谁小取谁,右边谁大取谁 [1,5]

2、具体步骤

  1. 第一行输入一个数字n,表示公共区间的数量;
  2. 接下来的n行,是具体的公共区间,并添加到集合list中;
  3. 两两组合求出公共区间,左右分别相比,谁小取谁,删除重复的公共区间;
    • 如果list就剩一个了,就不用比了;
    • 第一个数组 与 余下的数组进行两两比较;
    • 比较过的直接移除;
    • 遍历余下的数组;
      • 两个区间有交集;
      • 取交集,左取大,右取小;
      • 判断公共区间是否存在,如果不存在,加入到公共区间集合中;
    • 再取下一个第一个数组,再与余下的数组进行两两比较,循环往复;
  4. 有交集就合并,没交集就直接输出,左边谁小取谁,右边谁大取谁;
    • 如果list就剩一个了,就不用比了;
    • 第一个数组 与 余下的数组进行两两比较;
    • 此时不能直接删除,因为合并的才可以删除,不能合并的,直接输出即可;
    • 遍历余下的数组;
    • 当有交集时;
      • 左边谁小取谁,右边谁大取谁;
      • 删除有交集的区间;
      • 将合并后的区间加入到list,再进行比较合并;
      • 可以合并,重置mergeFlag为true,表示list中的数组还可以继续合并;
    • 如果当前比较的第一个数组,不能与余下的数组进行合并,将其删除;
      • 能与余下的数组进行合并的区间;
      • 可以合并,表示list中的数组还可以继续合并,重置合并表示为false;
    • 取第一个区间,与余下的区间进行合并,循环往复
  5. 按照左区间升序排序,如果相等,再按右区间升序排序;
  6. 将合并后的公共区间添加到builder中,输出即可。

五、Java算法源码

public class OdTest07 {
    // 公共区间集合
    static List<int[]> publicRangeList = new ArrayList<>();

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = Integer.valueOf(sc.nextLine());
        List<int[]> list = new ArrayList<>();
        for (int i = 0; i < n; i++) {
            int[] arr = Arrays.stream(sc.nextLine().split(" ")).mapToInt(Integer::parseInt).toArray();
            list.add(arr);
        }

        // 两两组合求出公共区间,左右分别相比,谁小取谁,删除重复的公共区间 [1,3][3,3][3,5]
        dfs(list);

        // 有交集就合并,没交集就直接输出,左边谁小取谁,右边谁大取谁 [1,5]
        mergeDfs(publicRangeList);

        publicRangeList.addAll(unMergeList);

        // 按照左区间升序排序,如果相等,再按右区间升序排序
        publicRangeList.sort((a, b) -> a[0] != b[0] ? a[0] - b[0] : a[1] - b[1]);

        StringBuilder builder = new StringBuilder();
        for (int i = 0; i < publicRangeList.size(); i++) {
            builder.append(publicRangeList.get(i)[0]).append(" ").append(publicRangeList.get(i)[1]).append("\n");
        }
        System.out.println(builder.deleteCharAt(builder.length() - 1));
    }

    // 两两组合求出公共区间
    private static void dfs(List<int[]> list) {
        // 如果list就剩一个了,就不用比了
        if (list.size() == 1) {
            return;
        }

        // 第一个数组 与 余下的数组进行两两比较
        int[] firstArr = list.get(0);
        int left = firstArr[0];
        int right = firstArr[1];

        // 比较过的直接移除
        list.remove(0);
        // 余下的数组
        for (int i = 0; i < list.size(); i++) {
            // 余下的数组
            int[] comareArr = list.get(i);
            // 两个区间有交集
            if (right >= comareArr[0] && comareArr[1] >= left) {
                // 取交集,左取大,右取小
                int compareLeft = left <= comareArr[0] ? comareArr[0] : left;
                int compareRight = right <= comareArr[1] ? right : comareArr[1];
                int[] range = new int[]{compareLeft, compareRight};
                // 判断公共区间是否存在,如果不存在,加入到公共区间集合中
                if (!compareArr(range)) {
                    publicRangeList.add(range);
                }
            }
        }
        // 再取下一个第一个数组,再与余下的数组进行两两比较,循环往复
        dfs(list);
    }

    // 判断公共区间是否存在
    private static boolean compareArr(int[] arr) {
        for (int i = 0; i < publicRangeList.size(); i++) {
            int[] rangeArr = publicRangeList.get(i);
            if (arr[0] == rangeArr[0] && arr[1] == rangeArr[1]) {
                return true;
            }
        }
        return false;
    }

    // 是否可以合并
    static boolean mergeFlag = false;
    // 不能合并的数组区间
    static List<int[]> unMergeList = new ArrayList<>();

    // 有交集就合并,没交集就直接输出,左边谁小取谁,右边谁大取谁
    private static void mergeDfs(List<int[]> list) {
        // 如果list就剩一个了,就不用比了
        if (list.size() == 1) {
            return;
        }

        // 第一个数组 与 余下的数组进行两两比较
        // [3,6][5,6][5,7][6,6][6,7][6,8][9,10]
        int[] firstArr = list.get(0);
        int left = firstArr[0];
        int right = firstArr[1];
        // 此时不能直接删除,因为合并的才可以删除,不能合并的,直接输出即可
        // 余下的数组
        for (int i = 1; i < list.size(); i++) {
            int[] comareArr = list.get(i);
            // [9,10][3,6][5,7][6,8]
            // 当有交集时
            if (right >= comareArr[0] && comareArr[1] >= left) {
                // 左边谁小取谁,右边谁大取谁
                int compareLeft = left <= comareArr[0] ? left : comareArr[0];
                int compareRight = right <= comareArr[1] ? comareArr[1] : right;
                int[] range = new int[]{compareLeft, compareRight};
                // 删除有交集的区间
                list.remove(firstArr);
                list.remove(comareArr);
                // 将合并后的区间加入到list,再进行比较合并
                list.add(range);
                // 可以合并,表示list中的数组还可以继续合并
                mergeFlag = true;
                break;
            }
        }

        // 如果当前比较的第一个数组,不能与余下的数组进行合并,将其删除
        if (!mergeFlag) {
            list.remove(firstArr);
            // 能与余下的数组进行合并的区间
            unMergeList.add(firstArr);
        } else {// 可以合并,表示list中的数组还可以继续合并
            // 重置合并表示为false
            mergeFlag = false;
        }
        // 取第一个区间,与余下的区间进行合并,循环往复
        mergeDfs(list);
    }
}

感觉这道题,不至于这么复杂吧。

再重新读一遍题目,看看能否优化一下~

因为最近一直在刷dfs的算法题,所以第一反应是采用dfs来解,不过,普通的for循环就可以解决了,确实简单了很多,找准方向才最重要~

核心思想没什么变化。

解题步骤也简化了很多。

  1. 第一行输入一个数字n,表示公共区间的数量;
  2. 接下来的n行,是具体的公共区间,并添加到集合list中;
  3. 按照左区间升序排序,如果相等,再按右区间升序排序;
  4. 定义公共区间集合publicRangeList;
  5. 遍历list,每一个数组与后面的数组分别比较,取交集;
    • 两个区间有交集,左右分别相比,取交集,左取大,右取小;
  6. 按照左区间升序排序,如果相等,再按右区间升序排序;
  7. 定义合并后的区间数组builder;
  8. 获取第一个有效的合并区间mergeArr;
  9. 遍历公共区间集合publicRangeList;
    • 有交集,取右区间的最大值;
    • 没有交集,拼接到合并的区间数组builder中;
    • 重置有效的合并区间为下一个区间;
  10. 输出合并后的区间数组即可。
public class OdTest07 {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = Integer.valueOf(sc.nextLine());
        List<int[]> list = new ArrayList<>();
        for (int i = 0; i < n; i++) {
            int[] arr = Arrays.stream(sc.nextLine().split(" ")).mapToInt(Integer::parseInt).toArray();
            list.add(arr);
        }

        // 按照左区间升序排序,如果相等,再按右区间升序排序
        list.sort((a, b) -> a[0] != b[0] ? a[0] - b[0] : a[1] - b[1]);

        // 公共区间集合
        List<int[]> publicRangeList = new ArrayList<>();

        // 遍历list,每一个数组与后面的数组分别比较,取交集
        for (int i = 0; i < list.size(); i++) {
            int[] arr = list.get(i);
            for (int j = i + 1; j < list.size(); j++) {
                int[] nextArr = list.get(j);
                // 两个区间有交集
                if (arr[1] >= nextArr[0]) {
                    // 左右分别相比,取交集,左取大,右取小
                    publicRangeList.add(new int[]{Math.max(arr[0], nextArr[0]), Math.min(arr[1], nextArr[1])});
                }
            }
        }

        // [3,6][5,6][6,6][5,7][6,8][6,7][9,10]
        if (publicRangeList.size() == 0) {
            System.out.println("None");
            return;
        }

        // [3,6][5,6][5,7][6,6][6,7][6,8][9,10]
        publicRangeList.sort((a, b) -> a[0] != b[0] ? a[0] - b[0] : a[1] - b[1]);

        // 合并后的区间数组
        StringBuilder builder = new StringBuilder();
        // 有效的合并区间
        int[] mergeArr = publicRangeList.get(0);
        for (int i = 1; i < publicRangeList.size(); i++) {
            int[] nextArr = publicRangeList.get(i);
            // 有交集,取右区间的最大值
            if (mergeArr[1] >= nextArr[0]) {
                mergeArr[1] = Math.max(mergeArr[1], nextArr[1]);
            } else {// 没有交集,拼接到合并的区间数组builder中
                builder.append(mergeArr[0]).append(" ").append(mergeArr[1]).append("\n");
                // 重置有效的合并区间为下一个区间
                mergeArr = nextArr;
            }
        }
        builder.append(mergeArr[0]).append(" ").append(mergeArr[1]).append("\n");
        // 输出合并后的区间数组
        System.out.println(builder.deleteCharAt(builder.length() - 1));
    }
}

六、效果展示

1、输入

5
9 10
5 7
6 11
3 8
3 6

2、输出

3 8
9 10

3、说明

  • 3 6和3 8的公共区间是3 6
  • 3 6和5 7的公共区间是5 6
  • 3 6和6 11的公共区间是6 6
  • 3 6和9 10的公共区间是无
  • 3 8和5 7的公共区间是5 7
  • 3 8和6 11的公共区间是6 8
  • 3 8和9 10的公共区间是无
  • 5 7和6 11的公共区间是6 7
  • 5 7和9 10的公共区间是无
  • 6 11和9 10的公共区间是9 10
  • 两两组合求出公共区间:[3,6][5,6][6,6][5,7][6,8][6,7][9,10]
  • 有交集就合并,没交集就直接输出:[3,8][9,10]

在这里插入图片描述


🏆下一篇:华为OD机试 - 最长的顺子 - 感谢@禁止你发言提供的更简便算法(Java 2023 B卷 200分)

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

刷的越多,抽中的概率越大,每一题都有详细的答题思路、详细的代码注释、样例测试,发现新题目,随时更新,全天CSDN在线答疑。

在这里插入图片描述

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

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

相关文章

【小呆的力学笔记】弹塑性力学的初步认知二:应力分析(1)

文章目录 1.1 一点的应力状态1.2 一点主应力状态1.3 应力偏张量、球张量、应力不变量 1.1 一点的应力状态 物体在受到外力或者自身不均匀的温度场等作用时&#xff0c;在其内部会产生内力&#xff0c;物体的内力与方向和截面都有关系。假设有一个受到外力作用的变形体&#xf…

【Linux】Linux线程概念和线程控制

文章目录 一、Linux线程概念1.什么是线程2.线程的优缺点3.线程异常4.线程用途5.Linux进程VS线程 二、线程控制1.线程创建2.线程终止3.线程等待4.线程分离 一、Linux线程概念 1.什么是线程 线程是进程内的一个执行流。 我们知道&#xff0c;一个进程会有对应的PCB&#xff0c;…

【LeetCode刷题笔记(12-1)】【Python】【有效的字母异位词】【排序/字符统计】【简单】

文章目录 引言有效的字母异位词题目描述提示 解决方案1&#xff1a;【排序】解决方案2&#xff1a;【字符统计】结束语 有效的字母异位词 引言 编写通过所有测试案例的代码并不简单&#xff0c;通常需要深思熟虑和理性分析。虽然这些代码能够通过所有的测试案例&#xff0c;但…

职场必备!这个微信管理效率神器一定要知道

很多小伙伴在管理微信时&#xff0c;都会遇到效率不高的问题&#xff0c;尤其是一些有多个微信号的人&#xff0c;如何高效管理微信成为了一道难题。 今天就给大家分享一款能高效管理微信的工具&#xff0c;让大家既能节省时间又可以提高工作效率&#xff01; 通过微信管理系…

Vue3-23-组件-依赖注入的使用详解

什么是依赖注入 个人的理解 &#xff1a; 依赖注入&#xff0c;是在 一颗 组件树中&#xff0c;由 【前代组件】 给 【后代组件】 提供 属性值的 一种方式 &#xff1b;这种方式 突破了 【父子组件】之间通过 props 的方式传值的限制&#xff0c;只要是 【前代组件】提供的 依…

自动化测试工具-Selenium:WebDriver的API/方法使用全解

我们上一篇文章介绍了Selenium的三大组件&#xff0c;其中介绍了WebDriver是最重要的组件。在这里&#xff0c;我们将看到WebDriver常用的API/方法&#xff08;注&#xff1a;这里使用Python语言来进行演示&#xff09;。 1. WebDriver创建 打开VSCode&#xff0c;我们首先引…

数据结构与算法之美学习笔记:39 | 回溯算法:从电影《蝴蝶效应》中学习回溯算法的核心思想

目录 前言如何理解“回溯算法”&#xff1f;两个回溯算法的经典应用内容小结 前言 本节课程思维导图&#xff1a; 我们在前面深度优先搜索算法利用的是回溯算法思想。这个算法思想非常简单&#xff0c;但是应用却非常广泛。它除了用来指导像深度优先搜索这种经典的算法设计之外…

原生微信小程序中使用-阿里字体图标-详解

步骤一 1、打开阿里巴巴矢量图标库 网址&#xff1a;iconfont-阿里巴巴矢量图标库 2、搜索字体图标&#xff0c;鼠标悬浮点击添加入库 3、按如下步骤添加到自己的项目 步骤二 进入微信开发者工具 1、创建 fonts文件夹 > iconfont.wxss 文件&#xff0c;将刚才的代码复制…

GLTF/GLB模型在线预览、编辑、动画查看以及材质修改

在线工具推荐&#xff1a; 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 GLTF在线编辑器提供了一个内置的模型查看器&#xff0c;可以加载和预…

NC65凭证保存时,报“错误:凭证内部错误号:[10001]凭证借贷金额不平!”

NC65凭证保存时&#xff0c;报“错误:凭证内部错误号&#xff1a;[10001]凭证借贷金额不平&#xff01;” 实际就是分录少录了2分钱。加上去即可。 代码排查&#xff1a; nc.bs.gl.voucher.VoucherBO.save(VoucherVO voucher, Boolean isneedcheck) throws BusinessExceptio…

微服务之服务注册与发现

服务注册发现 服务注册就是维护一个登记簿&#xff0c;它管理系统内所有的服务地址。当新的服务启动后&#xff0c;它会向登记簿交待自己的地址信息。服务的依赖方直接向登记簿要 Service Provider 地址就行了。当下用于服务注册的工具非常多 ZooKeeper&#xff0c;Consul&…

Centos8一键启动多个Springboot jar包 改进版

一、前言 上篇《Centos8一键启动多个Springboot jar包》写了在centos环境下如何快速启动多个jar包。实际使用下来还是会发现不够完美&#xff0c;如我想重新启动10个jar包里面的两个&#xff0c;我得这么写&#xff1a; ./start.sh restart test1.jar; ./start.sh restart te…

为什么项目管理工具需要项目财务信息?

在讨论项目时&#xff0c;钱是绕不开的话题。 资金是项目管理中最重要的因素之一&#xff0c;与范围和时间并列&#xff0c;三者共同构成了 “三重约束”。例如&#xff0c;如果缩短项目时间&#xff0c;就必须增加成本。 如果无法清楚了解开支及其对项目的影响&#xff08;反…

TrustZone之可信操作系统

有许多可信内核&#xff0c;包括商业和开源的。一个例子是OP-TEE&#xff0c;最初由ST-Ericsson开发&#xff0c;但现在是由Linaro托管的开源项目。OP-TEE提供了一个功能齐全的可信执行环境&#xff0c;您可以在OP-TEE项目网站上找到详细的描述。 OP-TEE的结构如下图所示&…

成熟又专业的内外网文件交换系统,了解一下!

近年来全球网络安全威胁态势的加速严峻&#xff0c;使得企业对于网络安全有了前所未有的关注高度。越来越多的企业在网络安全体系建设和日常工作中&#xff0c;都面临一个核心问题&#xff0c;如何保护企业核心数据资产&#xff1f; 绝大多数企业都在内部实施了网络隔离&#x…

WEB测试之兼容性测试

1. 软件兼容性测试 兼容性测试是指待测试项目在特定的硬件平台上&#xff0c;不同的应用软件之间&#xff0c;不同的操作系统平台上&#xff0c;在不同的网络等环境中能正常的运行的测试。 兼容性测试的目的&#xff1a;待测试项目在不同的操作系统平台上正常运行&#xff0c…

如何通过ETLCloud的API对接功能实现各种SaaS平台数据对接

前言 当前使用SaaS系统的企业越来越多&#xff0c;当我们需要对SaaS系统中产生的数据进行分析和对接时就需要与SaaS系统提供的API进行对接&#xff0c;因为SaaS一般是不会提供数据库表给企业&#xff0c;这时就应该使用ETL&#xff08;Extract, Transform, Load&#xff09;的…

数据挖掘-09-IBM员工流失率预测(包括数据和代码)

文章目录 0. 代码数据下载1. 背景描述2. 探索性数据分析2.1 数据集分布2.2 特征的相关性2.3 散点图可视化 3. 特征工程和分类编程4. 实施机器学习模型3.1 随机森林分类器a. 通过随机森林进行特征排名b. 可视化树形图 3.2 梯度增强分类器a. 基于梯度增强模型的特征排序 3.3 结论…

《每天一分钟学习C语言·四》文本读写操作及二进制读写

fopen(参数1,参数2)&#xff0c;第一个参数是要打开的文件名&#xff0c;文件名字符串&#xff0c;第二个参数用于制定文件打开模式的一个字符串。 注&#xff1a;如果要打开某个盘的文本如F盘test文件夹下的test.txt文本&#xff0c;则fopen(“F:\test\test.txt”,”r”); 程序…

【Spring实战】创建第一个项目

文章目录 使用 Spring Initializr 创建第一个项目1. 打开官网2. 填写信息3. 生成工程4. 解压工程5. 导入 IDEA6. 编写 Hello world7. 启动项目8. 访问验证9. 详细代码最后 Spring 是一个强大且广泛使用的 Java 开发框架&#xff0c;提供了全面的基础设施和工具&#xff0c;用于…