华为OD机试 - 部门人力分配 - 二分查找(Java 2024 D卷 200分)

在这里插入图片描述

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

专栏导读

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

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

一、题目描述

部门在进行需求开发时需要进行人力安排。当前部门需要完成 N 个需求,需求用 requirements[i] 表示,requirements[i] 表示第 i 个需求的工作量大小,单位:人月。这部分需求需要在 M 个月内完成开发,进行人力安排后每个月的人力是固定的。

目前要求每个月最多有 2 个需求开发,并且每个月需要完成的需求不能超过部门人力。请帮部门评估在满足需求开发进度的情况下,每个月需要的最小人力是多少?

二、输入描述

输入第一行为 M ,第二行为 requirements 。

M 表示需要开发时间要求,requirements 表示每个需求工作量大小

N 为 requirements 长度,1 ≤ N / 2 ≤ M ≤ N ≤ 10000,1 ≤ requirements[i]≤ 10^9

三、输出描述

对于每一组测试数据,输出部门需要人力需求,行末无多余的空格。

四、解题思路

本题要求我们在给定的开发时间(以月份为单位)内完成一系列需求开发,每个月最多完成2个需求,并且每个月需要的总人力(工作量)不能超过一个固定值。目标是找到每个月需要的最小人力。

为了实现这一目标,我们可以使用二分查找和贪心算法相结合的策略:

  1. 确定二分查找的范围:
    • 最小值 minCapacity:为单个需求的最大工作量,因为至少需要满足最重的需求。
    • 最大值 maxCapacity:为两个最重需求的工作量之和,这是最极端的情况下的最大人力需求。
  2. 二分查找:
    • 通过二分查找的方法,在 minCapacity 和 maxCapacity 之间找到最小的每月人力需求。
    • 每次取中间值 midCapacity,检查在这个人力值下是否可以在规定的 months 内完成所有需求。
    • 如果可以,说明当前人力值可能是一个解,但需要尝试更小的值以找到最优解;如果不可以,说明当前人力值太小,需要增加人力值。
  3. 检查可行性:
    • 使用贪心算法检查在给定的每月人力值下是否可以在规定的 months 内完成所有需求。
    • 从工作量最小的需求和工作量最大的需求开始,尽量将两个需求安排在同一个月内。如果不能,就安排一个需求。
    • 记录需要的月份数,并与规定的 months 进行比较。

五、Java算法源码

public class Test01 {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        // 读取开发时间要求的月份数
        int months = Integer.parseInt(scanner.nextLine());

        // 读取每个需求的工作量,并转换为整数数组
        int[] workload = Arrays.stream(scanner.nextLine().split(" "))
                .mapToInt(Integer::parseInt).toArray();

        // 输出计算得到的最小人力需求
        System.out.println(calculateMinManpower(months, workload));
    }

    /**
     * 计算满足开发进度的最小人力需求
     * @param months 需要开发的总月份数
     * @param workload 每个需求的工作量数组
     * @return 最小人力需求
     */
    public static long calculateMinManpower(int months, int[] workload) {
        Arrays.sort(workload); // 将需求工作量数组进行排序
        int numOfTasks = workload.length; // 获取需求的总数量

        // 如果只有一个需求,则返回该需求的工作量
        if (numOfTasks == 1) {
            return workload[0];
        }

        // 初始化二分查找的最小值和最大值
        long minCapacity = workload[numOfTasks - 1]; // 最小人力需求为最大工作量
        long maxCapacity = workload[numOfTasks - 2] + workload[numOfTasks - 1]; // 最大人力需求为最大的两个工作量之和
        long optimalCapacity = maxCapacity; // 初始最优人力需求设为最大值

        // 进行二分查找
        while (minCapacity <= maxCapacity) {
            long midCapacity = (minCapacity + maxCapacity) >> 1; // 计算中间值

            // 检查在当前中间值下是否能在规定月份内完成所有需求
            if (canCompleteWithinMonths(midCapacity, months, workload)) {
                optimalCapacity = midCapacity; // 更新最优人力需求
                maxCapacity = midCapacity - 1; // 尝试更小的值
            } else {
                minCapacity = midCapacity + 1; // 尝试更大的值
            }
        }

        return optimalCapacity; // 返回计算得到的最优人力需求
    }

    /**
     * 检查是否可以在给定的月份内完成所有需求
     * @param capacity 每个月的最大工作量
     * @param months 总月份数
     * @param workload 每个需求的工作量数组
     * @return 是否可以在给定的月份内完成所有需求
     */
    public static boolean canCompleteWithinMonths(long capacity, int months, int[] workload) {
        int left = 0; // 指向工作量最小的需求
        int right = workload.length - 1; // 指向工作量最大的需求
        int requiredMonths = 0; // 记录需要的月份数

        // 遍历需求,尝试将两个需求放在同一个月份内完成
        while (left <= right) {
            if (workload[left] + workload[right] <= capacity) {
                left++; // 如果最小需求和最大需求可以一起完成,则移动左指针
            }
            right--; // 总是移动右指针
            requiredMonths++; // 增加需要的月份数
        }

        return months >= requiredMonths; // 判断给定的月份是否足够
    }
}

六、效果展示

1、输入

3
3 5 3 4

2、输出

6

3、说明

输入数据两行,第一行输入数据 3 表示开发时间要求,第二行输入数据表示需求工作量大小,输出数据一行,表示部门人力需求。

当选择人力为6时,2个需求量为3的工作可以在1个月里完成,其他2个工作各需要1个月完成。可以在3个月内完成所有需求。

当选择人力为5时,4个工作各需要1个月完成,一共需要4个月才能完成所有需求。

因此6是部门最小的人力需求。

在这里插入图片描述


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

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

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

在这里插入图片描述

  • 21
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
华为OD-2023真题主要考察了以下几个方面的知识点: 1. 数据结构与算法:题目涉及了常见的数据结构和算法,如数组、链表、树、图、排序、搜索等。要求考生熟悉这些数据结构的基本操作和常用算法的实现方法。 2. 编程语言:题目要求使用C++或Java语言完成编程任务,要求考生熟悉相应语言的语法和常用的库函数使用方法。 3. 网络通信:题目涉及了网络通信相关的知识点,如TCP/IP协议、HTTP协议、socket编程等。要求考生了解网络通信的基本概念和实现方式。 4. 操作系统:题目要求考生熟悉操作系统相关的知识,如进程管理、内存管理、文件系统等。要求考生了解操作系统的基本功能和实现原理。 5. 数据库:题目涉及了数据库相关的知识点,如SQL语句的编写、数据库的设计和优化等。要求考生具备数据库的基本操作和设计能力。 6. 设计模式:题目要求考生能够根据给定的需求设计相应的对象和类,并且符合设计模式的原则。要求考生熟悉常见的设计模式和其应用场景。 7. 系统设计与架构:题目要求考生从整体上设计和实现一个系统,并考虑系统的性能、可扩展性等因素。要求考生了解系统设计的基本原则和方法。 以上是华为OD-2023真题的一些考点分类。考生在复习备考时,可以根据这些考点有针对性地进行学习和练习,提升自己的应能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

哪 吒

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值