友情支持

如果您觉得这个笔记对您有所帮助,看在D瓜哥码这么多字的辛苦上,请友情支持一下,D瓜哥感激不尽,😜

支付宝

微信

有些打赏的朋友希望可以加个好友,欢迎关注D 瓜哥的微信公众号,这样就可以通过公众号的回复直接给我发信息。

wx jikerizhi

公众号的微信号是: jikerizhi因为众所周知的原因,有时图片加载不出来。 如果图片加载不出来可以直接通过搜索微信号来查找我的公众号。

621. 任务调度器

给你一个用字符数组 tasks 表示的 CPU 需要执行的任务列表,用字母 A 到 Z 表示,以及一个冷却时间 n。每个周期或时间间隔允许完成一项任务。任务可以按任何顺序完成,但有一个限制:两个 相同种类 的任务之间必须有长度为 n 的冷却时间。

返回完成所有任务所需要的 最短时间间隔

示例 1:

输入:tasks = ["A","A","A","B","B","B"], n = 2

输出:8

解释:

在完成任务 A 之后,你必须等待两个间隔。对任务 B 来说也是一样。在第 3 个间隔,A 和 B 都不能完成,所以你需要待命。在第 4 个间隔,由于已经经过了 2 个间隔,你可以再次执行 A 任务。

示例 2:

输入:tasks = ["A","C","A","B","D","B"], n = 1

输出:6

解释:一种可能的序列是:A → B → C → D → A → B。

由于冷却间隔为 1,你可以在完成另一个任务后重复执行这个任务。

示例 3:

输入:tasks = ["A","A","A","B","B","B"], n = 3

输出:10

解释:一种可能的序列为:A → B → idle → idle → A → B → idle → idle → A → B。

只有两种任务类型,A 和 B,需要被 3 个间隔分割。这导致重复执行这些任务的间隔当中有两次待命状态。

提示:

  • 1 <= tasks.length <= 104

  • tasks[i] 是大写英文字母

  • 0 <= n <= 100

思路分析

这道题有三个点需要注意:

  1. 因为要任务休息时间,所以,出现次数最多的任务,会持续得更长,有将任务按照出现次数排序,优先安排次数多的任务。

  2. 结果关注的是总共需要完成的时间,所以不需要关注具体执行的是哪个任务。

  3. 需要"空转时间"的处理。

解题思路:

  1. 将任务按类型分组,正好A-Z用一个int[26]保存任务类型个数

  2. 对数组进行排序,优先排列个数(count)最大的任务,如题得到的时间至少为 retCount =(count-1)* (n+1) + 1 =⇒ A→X→X→A→X→X→A(X为其他任务或者待命)

  3. 再排序下一个任务,如果下一个任务B个数和最大任务数一致,则retCount++ =⇒ A→B→X→A→B→X→A→B

  4. 如果空位都插满之后还有任务,那就随便在这些间隔里面插入就可以,因为间隔长度肯定会大于n,在这种情况下就是任务的总数是最小所需时间

  • 一刷

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
/**
 * Runtime: 2 ms, faster than 99.97% of Java online submissions for Task Scheduler.
 * Memory Usage: 42.7 MB, less than 5.88% of Java online submissions for Task Scheduler.
 *
 * Copy from: https://leetcode-cn.com/problems/task-scheduler/solution/621-ren-wu-diao-du-qi-java-jie-ti-zhu-shi-ying-gai/[621. 任务调度器--Java--解题注释应该能看懂 - 任务调度器 - 力扣(LeetCode)]
 *
 * @author D瓜哥 · https://www.diguage.com
 * @since 2020-01-31 21:26
 */
public int leastInterval(char[] tasks, int n) {
    if (Objects.isNull(tasks) || tasks.length == 0) {
        return 0;
    }
    int[] counts = new int[26];
    for (char task : tasks) {
        counts[task - 'A']++;
    }
    Arrays.sort(counts);
    int maxCount = counts[25];
    // 前面 maxCount - 1 个任务需要有 n 个间隔,所以就是 (maxCount - 1) * (n + 1)
    // 而最后一个任务不需要间隔,就只能算 1
    int restCount = (maxCount - 1) * (n + 1) + 1;
    for (int i = 24; i >= 0 && counts[i] == maxCount; i--) {
        restCount++;
    }
    return Math.max(restCount, tasks.length);
}

/**
 * Runtime: 2 ms, faster than 99.97% of Java online submissions for Task Scheduler.
 * Memory Usage: 42.6 MB, less than 5.88% of Java online submissions for Task Scheduler.
 * <p>
 * Copy from: https://leetcode-cn.com/problems/task-scheduler/solution/ren-wu-diao-du-qi-by-leetcode/[任务调度器 - 任务调度器 - 力扣(LeetCode)]
 */
public int leastInterval1(char[] tasks, int n) {
    if (Objects.isNull(tasks) || tasks.length == 0) {
        return 0;
    }
    int[] counts = new int[26];
    for (char task : tasks) {
        counts[task - 'A']++;
    }
    Arrays.sort(counts);
    int maxValue = counts[25] - 1;
    int idleSlots = maxValue * n;
    for (int i = 24; i >= 0 && counts[i] > 0; i--) {
        idleSlots -= Math.min(maxValue, counts[i]);
    }
    return idleSlots > 0 ? idleSlots + tasks.length : tasks.length;
}

/**
 * Runtime: 29 ms, faster than 42.49% of Java online submissions for Task Scheduler.
 * Memory Usage: 41.9 MB, less than 5.88% of Java online submissions for Task Scheduler.
 * <p>
 * Copy from: https://leetcode-cn.com/problems/task-scheduler/solution/ren-wu-diao-du-qi-by-leetcode/[任务调度器 - 任务调度器 - 力扣(LeetCode)]
 */
public int leastIntervalQueue(char[] tasks, int n) {
    if (Objects.isNull(tasks) || tasks.length == 0) {
        return 0;
    }
    int[] counts = new int[26];
    for (char c : tasks) {
        counts[c - 'A']++;
    }
    PriorityQueue<Integer> queue = new PriorityQueue<>(Collections.reverseOrder());
    for (int count : counts) {
        if (count > 0) {
            queue.add(count);
        }
    }
    int time = 0;
    while (!queue.isEmpty()) {
        List<Integer> temp = new ArrayList<>();
        int i = 0;
        while (i <= n) {
            if (!queue.isEmpty()) {
                if (queue.peek() > 1) {
                    temp.add(queue.poll() - 1);
                } else {
                    queue.poll();
                }
            }
            time++;
            i++;
            if (queue.isEmpty() && temp.isEmpty()) {
                break;
            }
        }
        for (Integer count : temp) {
            queue.add(count);
        }
    }
    return time;
}

参考资料