We use cookies (including Google cookies) to personalize ads and analyze traffic. By continuing to use our site, you accept our Privacy Policy.

Process Tasks Using Servers

Difficulty: Medium


Problem Description

You are given two 0-indexed integer arrays servers and tasks of lengths n and m respectively. servers[i] is the weight of the i-th server, and tasks[j] is the time needed to process the j-th task in seconds. Tasks are assigned to the servers using a task queue. Initially, all servers are free, and the queue is empty. At second j, the j-th task is inserted into the queue. As long as there are free servers and the queue is not empty, the task in the front of the queue will be assigned to a free server with the smallest weight, and in case of a tie, it is assigned to a free server with the smallest index. If there are no free servers and the queue is not empty, we wait until a server becomes free and immediately assign the next task. If multiple servers become free at the same time, then multiple tasks from the queue will be assigned in order of insertion following the weight and index priorities above. A server that is assigned task j at second t will be free again at second t + tasks[j]. Build an array ans of length m, where ans[j] is the index of the server the j-th task will be assigned to. Return the array ans.


Key Insights

  • Use a min-heap (priority queue) to manage free servers based on weight and index.
  • Use another min-heap to manage tasks waiting for server availability.
  • Track when servers become free and assign tasks accordingly.
  • Ensure tasks are processed in the order they arrive, respecting server availability.

Space and Time Complexity

Time Complexity: O((n + m) log n)
Space Complexity: O(n + m)


Solution

We will utilize two priority queues (min-heaps): one for the servers and one for the tasks. The server heap will prioritize servers based on their weight, and in case of ties, by their index. As tasks arrive, they will be added to the task queue. When a server becomes free, we will assign the next available task from the task queue to the server. This process will continue until all tasks are assigned.


Code Solutions

import heapq

def assignTasks(servers, tasks):
    n, m = len(servers), len(tasks)
    free_servers = [(weight, index) for index, weight in enumerate(servers)]
    heapq.heapify(free_servers)
    
    busy_servers = []
    ans = [0] * m
    time = 0
    
    for task_index in range(m):
        # Add the task to the time
        time = max(time, task_index)
        heapq.heappush(busy_servers, (time + tasks[task_index], task_index))
        
        # Assign tasks to available servers
        while free_servers and busy_servers:
            if time >= busy_servers[0][0]:  # If a server is free
                _, busy_task_index = heapq.heappop(busy_servers)
                weight, server_index = heapq.heappop(free_servers)
                ans[busy_task_index] = server_index
                time += tasks[busy_task_index]  # Update time to when the server gets free
            else:
                break
        
        # If no free servers, wait for the next server to be free
        if not free_servers and busy_servers:
            time = busy_servers[0][0]

        # Add a free server back to the heap if it has become available
        while busy_servers and busy_servers[0][0] <= time:
            _, busy_task_index = heapq.heappop(busy_servers)
            weight, server_index = heapq.heappop(free_servers)
            ans[busy_task_index] = server_index
    
    return ans
← Back to All Questions