Leetcode 第 336 场周赛 Problem D 完成所有任务的最少时间

题目:

   2589. 完成所有任务的最少时间

   你有一台电脑,它可以 同时 运行无数个任务。给你一个二维整数数组 tasks ,其中 tasks[i] = [starti, endi, durationi]

   表示第 i 个任务需要在 闭区间 时间段 [starti, endi] 内运行 durationi 个整数时间点(但不需要连续)。

   当电脑需要运行任务时,你可以打开电脑,如果空闲时,你可以将电脑关闭。

   请你返回完成所有任务的情况下,电脑最少需要运行多少秒。

   1 <= tasks.length <= 2000

   tasks[i].length == 3

   1 <= starti, endi <= 2000

   1 <= durationi <= endi - starti + 1

解法:

  每个任务按照右端点升序,然后循环每个任务,保证运行时间足够的情况下先让后缀运行,这样能与后面的任务重叠的可能性更高,

  最简单的是开一个标记数组,代表当前时间点正在运行,循环任务数组同时循环标记数组即可

  U 代表 end 的最大值,时间复杂度:O(nU),空间复杂度:O(U)

  

代码:

    /**
     * 每个任务按照右端点升序,然后循环每个任务,保证运行时间足够的情况下先让后缀运行,这样能与后面的任务重叠的可能性更高,
     * 最简单的是开一个标记数组,代表当前时间点正在运行,循环任务数组同时循环标记数组即可
     * U 代表 end 的最大值,时间复杂度:O(nU),空间复杂度:O(U)
     */
    private int solution(int[][] tasks) {
        if (tasks == null || tasks.length == 0) {
            return 0;
        }
        int res = 0;
        // 右端点升序任务数组
        Arrays.sort(tasks, (t1, t2) -> {return t1[1] - t2[1];});
//        Arrays.stream(tasks).forEach(t -> {System.out.println(t[0] + ":" + t[1] + ":" + t[2]);});
        // 标记数组长度为最大的时间(最后一个节点的结束时间)
        boolean[] flag = new boolean[tasks[tasks.length - 1][1] + 1];
        // 循环任务数组
        for (int i = 0; i < tasks.length; i++) {
            int duration = tasks[i][2];
            // 已标记过代表已经执行过了
            for (int j = tasks[i][1]; j >= tasks[i][0] && duration > 0; j--) {
                if (flag[j]) {
                    duration--;
                }
            }
            // 如果未达到需要执行的时间点,从该任务后缀开始更新标记数组
            if(duration > 0) {
                for (int j = tasks[i][1]; j >= tasks[i][0] && duration > 0; j--) {
                    if (!flag[j]) {
                        duration--;
                        res++;
                        flag[j] = true;
                    }
                }
            }
        }
        return res;
    }

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值