JavaInterview JavaInterview
首页
指南
分类
标签
归档
  • CSDN (opens new window)
  • 文档集合 (opens new window)
  • 系统架构 (opens new window)
  • 微信号 (opens new window)
  • 公众号 (opens new window)

『Java面试+Java学习』
首页
指南
分类
标签
归档
  • CSDN (opens new window)
  • 文档集合 (opens new window)
  • 系统架构 (opens new window)
  • 微信号 (opens new window)
  • 公众号 (opens new window)
  • 指南
  • 简历

  • Java

  • 面试

  • 算法

  • algorithm
  • leetcode
JavaInterview.cn
2025-06-03
目录

1606. 找到处理最多请求的服务器Java

文章发布较早,内容可能过时,阅读注意甄别。

# 题目

你有 k 个服务器,编号为 0 到 k-1 ,它们可以同时处理多个请求组。每个服务器有无穷的计算能力但是 不能同时处理超过一个请求 。请求分配到服务器的规则如下:

  • 第 i (序号从 0 开始)个请求到达。
  • 如果所有服务器都已被占据,那么该请求被舍弃(完全不处理)。
  • 如果第 (i % k) 个服务器空闲,那么对应服务器会处理该请求。
  • 否则,将请求安排给下一个空闲的服务器(服务器构成一个环,必要的话可能从第 0 个服务器开始继续找下一个空闲的服务器)。比方说,如果第 i 个服务器在忙,那么会查看第 (i+1) 个服务器,第 (i+2) 个服务器等等。
  • 给你一个 严格递增 的正整数数组 arrival ,表示第 i 个任务的到达时间,和另一个数组 load ,其中 load[i] 表示第 i 个请求的工作量(也就是服务器完成它所需要的时间)。你的任务是找到 最繁忙的服务器 。最繁忙定义为一个服务器处理的请求数是所有服务器里最多的。

请你返回包含所有 最繁忙服务器 序号的列表,你可以以任意顺序返回这个列表。

示例 1: load-1.png

输入:k = 3, arrival = [1,2,3,4,5], load = [5,2,3,3,3]
输出:[1]
解释:
所有服务器一开始都是空闲的。
前 3 个请求分别由前 3 台服务器依次处理。
请求 3 进来的时候,服务器 0 被占据,所以它被安排到下一台空闲的服务器,也就是服务器 1 。
请求 4 进来的时候,由于所有服务器都被占据,该请求被舍弃。
服务器 0 和 2 分别都处理了一个请求,服务器 1 处理了两个请求。所以服务器 1 是最忙的服务器。

示例 2:

输入:k = 3, arrival = [1,2,3,4], load = [1,2,1,2]
输出:[0]
解释:
前 3 个请求分别被前 3 个服务器处理。
请求 3 进来,由于服务器 0 空闲,它被服务器 0 处理。
服务器 0 处理了两个请求,服务器 1 和 2 分别处理了一个请求。所以服务器 0 是最忙的服务器。

示例 3:

输入:k = 3, arrival = [1,2,3], load = [10,12,11]
输出:[0,1,2]
解释:每个服务器分别处理了一个请求,所以它们都是最忙的服务器。

示例 4:

输入:k = 3, arrival = [1,2,3,4,8,9,10], load = [5,2,10,3,1,2,2]
输出:[1]

示例 5:

输入:k = 1, arrival = [1], load = [1]
输出:[0]

提示:

  • 1 <= k <= 105
  • 1 <= arrival.length, load.length <= 105
  • arrival.length == load.length
  • 1 <= arrival[i], load[i] <= 109
  • arrival 保证 严格递增 。

# 思路

找到环中合适的空闲位置

# 解法

class Solution {
    public List<Integer> busiestServers(int k, int[] arrival, int[] load) {
        List<Integer> res = new ArrayList<>();

        TreeSet<Integer> freeServer = new TreeSet<>();

        // int[2]: [结束时间, 服务器编号]
        PriorityQueue<int[]> priorityQueue = new PriorityQueue<>((a, b) -> (a[0] - b[0]));

        for (int i = 0; i < k; i++) {
            freeServer.add(i);
        }

        int taskNum = arrival.length;

        int idx = 0;

        // 各服务处理数量
        int[] serverDeals = new int[k];

        while (idx < taskNum) {
            // 服务器编号
            int sNo = idx % k;

            // 任务开始时间
            int startTime = arrival[idx];

            // 释放忙完服务器 && 计数服务器处理任务数
            while (!priorityQueue.isEmpty()) {
                int[] e = priorityQueue.peek();
                if (e[0] <= startTime) { // 若已处理任务结束时间 <= 任务起始时间
                    priorityQueue.poll(); // 弹出
                    freeServer.add(e[1]); // 将机器加入空闲列表
                    serverDeals[e[1]]++;
                } else { // 否则, 跳出
                    break;
                }
            }

            // 若执行释放操作后, 仍无可用服务器; 直接跳过当前任务
            if (freeServer.isEmpty()) {
                idx++;
                continue;
            }

            // 找到环中合适的空闲位置
//            while (!freeServer.contains(sNo)) {
//                if (sNo == k) {
//                    sNo = 0;
//                } else {
//                    sNo++;
//                }
//            }
            Integer nextNo = freeServer.ceiling(sNo);

            nextNo = nextNo == null ? freeServer.ceiling(0): nextNo;
            
            // 将对应服务器移除空闲列表
            freeServer.remove(nextNo);

            // 编号为 idx 的任务的结束时间
            int endTime = arrival[idx] + load[idx];

            // 将 <新任务结束时间, 服务器> 加入队列
            priorityQueue.add(new int[]{endTime, nextNo});
            idx++;
        }

        while (!priorityQueue.isEmpty()) {
            int sNo = priorityQueue.poll()[1];
            serverDeals[sNo]++;
        }

        int max = 0;
        for (int serverDeal : serverDeals) {
            max = Math.max(max, serverDeal);
        }

        for (int i = 0; i < serverDeals.length; i++) {
            if (serverDeals[i] == max) {
                res.add(i);
            }
        }

        return res;
    }
}

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

# 总结

  • 分析出几种情况,然后分别对各个情况实现
微信 支付宝
#Java
最近更新
01
1633. 各赛事的用户注册率 Java
07-13
02
1636. 按照频率将数组升序排序 Java
07-13
03
1640. 能否连接形成数组 Java
07-13
更多文章>
Theme by Vdoing | Copyright © 2019-2025 JavaInterview.cn
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式