1029. Two City Scheduling


Problem Description

A company is conducting interviews and needs to fly interviewees to two different cities, city a and city b. They have a total of 2n people to interview and need to send exactly n people to each city. The cost of flying each person to each city is different and is provided in a list known as costs. The costs list is structured such that costs[i] = [aCost_i, bCost_i], with aCost_i representing the cost to fly the ith person to city a and bCost_i representing the cost to fly the ith person to city b. The goal is to determine the minimum possible total cost to fly exactly n people to city a and n people to city b.

Intuition

The intuition behind the solution is to use a greedy strategy. A greedy algorithm makes the locally optimal choice at each step with the hope of finding the global optimum. In the context of this problem, we want to minimize the total cost of flying people to the two cities. To do this, we need to find the n people for whom the difference between flying to city a and city b is the largest in favor of city a, and fly them there. This will ensure that we are saving as much money as possible on the first n flights.

By sorting the costs array based on the difference aCost_i - bCost_i in ascending order, we can easily find the n people to fly to city a (those at the start of the sorted list) and the n people to fly to city b (those at the end of the list). This sorting allows us to consider the relative costs of flying each person to each city and ensure that, overall, we are making the most cost-effective decisions.

The solution handles the sorting and then computes the total cost by summing up the costs for flying the first n people to city a and the last n people to city b.

Learn more about Greedy and Sorting patterns.

Solution Approach

The solution approach for the given problem uses a greedy algorithm, which is a common pattern for optimization problems. The idea here is to make a sequence of choices that are locally optimal in order to reach a solution that is close to or equal to the optimal global solution. Here's a breakdown of how the approach is implemented:

Steps Involved:

  1. Sorting Costs: First, we sort the costs array by the difference in cost between flying to city a and city b (aCost_i - bCost_i). By doing this, we cluster people who are cheaper to fly to city a at the beginning of the array and those cheaper to city b towards the end.

    costs.sort(key=lambda x: x[0] - x[1])

  2. Calculating Half Length: Since we need to send n people to each city and we have 2n people in total, we find n by dividing the length of the costs array by 2. Note that we're using a bit shift operation to divide by 2, which is equivalent but slightly faster in most programming languages.

    n = len(costs) >> 1

  3. Determining Minimum Cost: Finally, we take the sum of the costs for flying the first n people to city a and the last n people to city b, which gives us the minimum cost.

    sum(costs[i][0] + costs[i + n][1] for i in range(n))

Data Structures Used:

  • A list of lists is used to store the flying costs to both cities for each person.

Algorithms & Patterns Used:

  • Greedy Algorithm: The algorithm sorts the array and then selects the first n elements for one city and the last n elements for the other city, based on the local optimality of the cost difference.
  • Sorting: A key step in the greedy approach where the array is sorted based on a specific condition (cost difference here).
  • List Comprehension: Python list comprehension is used to sum the costs which result in a concise and readable implementation.

Complexity Analysis:

  • The time complexity of the solution is dominated by the sorting step, which is O(n log n) where n is the number of people.
  • The space complexity is O(1) or constant space, aside from the input, since no additional data structures are used that grow with the size of the input.

The algorithm implemented here effectively balances the costs for both cities by understanding and utilizing the differences in individual costs to minimize the total expenditure.

Ready to land your dream job?

Unlock your dream job with a 2-minute evaluator for a personalized learning plan!

Start Evaluator

Example Walkthrough

Let's consider an example where a company has 2n = 4 people (n = 2) to send to cities a and b. The costs for flying each person to cities a and b are provided as follows:

1costs = [[400, 200], [300, 600], [100, 500], [200, 300]]

Now, let's walk through the solution approach step by step using this example:

  1. Sorting Costs First, we sort the costs array based on the difference between flying to city a and city b, so we calculate aCost_i - bCost_i for each person:

    1Person 1: 400 - 200 = 200
    2Person 2: 300 - 600 = -300
    3Person 3: 100 - 500 = -400
    4Person 4: 200 - 300 = -100

    After sorting these based on the difference in ascending order, our costs array looks like this:

    1costs = [[100, 500], [300, 600], [200, 300], [400, 200]]

    Now, the first two people on the list are the ones we will fly to city a, and the last two to city b.

  2. Calculating Half Length Here, n is already given as 2 (since 2n = 4). In a more general sense, we would compute n as len(costs) >> 1.

  3. Determining Minimum Cost Lastly, we will add the cost for flying the first n people to city a and the last n people to city b:

    1For city a: costs[0][0] + costs[1][0] = 100 + 300 = 400
    2For city b: costs[2][1] + costs[3][1] = 300 + 200 = 500

    The total minimum cost to fly all these people is the sum of the costs for city a and city b: 400 + 500 = 900.

Through this example, we can see how the company efficiently reduces the total cost for flying out interviewees to two different cities by implementing a greedy algorithm that focuses on sending each person to the city that results in the lowest additional cost.

Solution Implementation

1from typing import List
2
3class Solution:
4    def two_city_sched_cost(self, costs: List[List[int]]) -> int:
5        # Sort the costs list based on the cost difference between the two cities (A and B).
6        # This helps in figuring out the cheaper city for each person in a way that can minimize the total cost.
7        costs.sort(key=lambda cost: cost[0] - cost[1])
8      
9        # Compute the number of people that should go to each city.
10        # Since it's a two-city schedule, half the people will go to each city.
11        num_people = len(costs) // 2  # Using // for floor division which is standard in Python 3.
12      
13        # Calculate the total minimum cost by adding the cost of sending the first half
14        # of the people to city A and the remaining half to city B.
15        # This works because the costs array is now sorted in a way that the first half
16        # are the people who have a cheaper cost going to city A compared
17        # to city B, and vice versa for the second half.
18        total_cost = sum(costs[i][0] for i in range(num_people)) + \
19                     sum(costs[i + num_people][1] for i in range(num_people))
20      
21        return total_cost
22
23# Example usage:
24# costs = [[10,20],[30,200],[400,50],[30,20]]
25# solution = Solution()
26# print(solution.two_city_sched_cost(costs))  # Should print the result of the cost calculation based on the provided costs array.
27
1import java.util.Arrays; // Import the Arrays class to use the sort method
2
3class Solution {
4    public int twoCitySchedCost(int[][] costs) {
5        // Sort the array based on the cost difference between the two cities (a and b)
6        // for each person. The comparator subtracts the cost of going to city B from the
7        // cost of going to city A for each person, and sorts based on these differences.
8        Arrays.sort(costs, (a, b) -> (a[0] - a[1]) - (b[0] - b[1]));
9
10        int totalCost = 0; // Initialize total cost to 0
11        int n = costs.length / 2; // Calculate n, where n is half the number of people (half will go to city A, half to city B)
12
13        // Accumulate the minimum cost by sending the first n people to city A
14        // and the remaining n people to city B based on the sorted order
15        for (int i = 0; i < n; ++i) {
16            totalCost += costs[i][0];   // Add cost for city A
17            totalCost += costs[i + n][1]; // Add cost for city B
18        }
19
20        return totalCost; // Return the accumulated total cost
21    }
22}
23
1#include <vector>
2#include <algorithm> // Include necessary headers
3
4class Solution {
5public:
6    // Function to calculate the minimum cost to send n people to two cities.
7    int twoCitySchedCost(vector<vector<int>>& costs) {
8        // Sort the input vector based on the cost difference for attending city A and city B
9        sort(costs.begin(), costs.end(), [](const vector<int>& a, const vector<int>& b) {
10            return a[0] - a[1] < b[0] - b[1];
11        });
12
13        // The total number of people is twice the number of people we need to send to one city
14        int totalPeople = costs.size();
15        // Calculate the number of people to send to each city
16        int halfPeople = totalPeople / 2;
17        // Initialize answer to store the total minimized cost
18        int totalCost = 0;
19
20        // Calculate the minimum total cost
21        for (int i = 0; i < halfPeople; ++i) {
22            // Add up the cost of sending the first half of people to city A
23            totalCost += costs[i][0];
24            // Add up the cost of sending the second half of people to city B
25            totalCost += costs[i + halfPeople][1];
26        }
27
28        // Return the computed total cost
29        return totalCost;
30    }
31};
32
1function twoCitySchedCost(costs: number[][]): number {
2    // Sort the array of costs based on the difference in cost between sending 
3    // a person to city A and city B.
4    costs.sort((firstPair, secondPair) => (firstPair[0] - firstPair[1]) - (secondPair[0] - secondPair[1]));
5
6    // Calculate half the length of the costs array since we're splitting the group in two
7    const halfLength = costs.length / 2;
8
9    // Initialize an accumulator for the total cost
10    let totalCost = 0;
11
12    // Iterate over the first half of the sorted array for city A costs and the second 
13    // half for city B costs to compute the minimum total cost
14    for (let i = 0; i < halfLength; ++i) {
15        // Add the cost of sending the i-th person to city A from the first half
16        totalCost += costs[i][0];
17
18        // Add the cost of sending the i-th person to city B from the second half
19        totalCost += costs[i + halfLength][1];
20    }
21
22    // Return the computed total cost
23    return totalCost;
24}
25

Time and Space Complexity

Time Complexity

The time complexity of the provided code consists of the following parts:

  1. Sorting the costs list: Since the sort() function in Python uses the Timsort algorithm, which has a time complexity of O(n log n), where n is the total number of elements in the list. Here n is 2N since we're supposed to send 2N people to two cities (N to each city).

  2. Summing up the costs: The list comprehension iterates over the first half and the second half of the sorted list, resulting in N iterations (since n = len(costs) >> 1 computes to N). The summing operation inside the list comprehension is O(1) for each element, so the overall time for this part is O(N).

Combining these two parts, the total time complexity is O(n log n) + O(N), which simplifies to O(n log n) since logarithmic factors are dominant for large n.

Hence, the time complexity of the code is O(n log n).

Space Complexity

The space complexity of the code analysis involves:

  1. The sorted list costs: Sorting is done in-place, so it does not consume additional space proportional to the input size.

  2. The list comprehension for summing the costs and the range used within: both utilize only a small constant amount of additional space.

Therefore, the space complexity of the entire operation is O(1), as no significant additional space is used that would increase with the size of the input.

Overall, the space complexity of the code is O(1).

Learn more about how to find time and space complexity quickly using problem constraints.


Discover Your Strengths and Weaknesses: Take Our 2-Minute Quiz to Tailor Your Study Plan:
Question 1 out of 10

What does the following code do?

1def f(arr1, arr2):
2  i, j = 0, 0
3  new_arr = []
4  while i < len(arr1) and j < len(arr2):
5      if arr1[i] < arr2[j]:
6          new_arr.append(arr1[i])
7          i += 1
8      else:
9          new_arr.append(arr2[j])
10          j += 1
11  new_arr.extend(arr1[i:])
12  new_arr.extend(arr2[j:])
13  return new_arr
14
1public static List<Integer> f(int[] arr1, int[] arr2) {
2  int i = 0, j = 0;
3  List<Integer> newArr = new ArrayList<>();
4
5  while (i < arr1.length && j < arr2.length) {
6      if (arr1[i] < arr2[j]) {
7          newArr.add(arr1[i]);
8          i++;
9      } else {
10          newArr.add(arr2[j]);
11          j++;
12      }
13  }
14
15  while (i < arr1.length) {
16      newArr.add(arr1[i]);
17      i++;
18  }
19
20  while (j < arr2.length) {
21      newArr.add(arr2[j]);
22      j++;
23  }
24
25  return newArr;
26}
27
1function f(arr1, arr2) {
2  let i = 0, j = 0;
3  let newArr = [];
4  
5  while (i < arr1.length && j < arr2.length) {
6      if (arr1[i] < arr2[j]) {
7          newArr.push(arr1[i]);
8          i++;
9      } else {
10          newArr.push(arr2[j]);
11          j++;
12      }
13  }
14  
15  while (i < arr1.length) {
16      newArr.push(arr1[i]);
17      i++;
18  }
19  
20  while (j < arr2.length) {
21      newArr.push(arr2[j]);
22      j++;
23  }
24  
25  return newArr;
26}
27

Recommended Readings

Got a question?ย Ask the Monster Assistantย anything you don't understand.

Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.

Coding Interview Strategies

Dive into our free, detailed pattern charts and company guides to understand what each company focuses on.

See Patterns
โ†
โ†‘๐Ÿช„