2187. Minimum Time to Complete Trips
Problem Description
In this problem, we are given an array time
that represents the time each bus takes to complete a single trip. Importantly, the buses can each make as many trips as needed back-to-back, without any downtime in between successive trips. Moreover, the buses operate independently from one another, meaning that the trips one bus makes have no effect on another's trips.
We also receive an integer totalTrips
, which signifies the collective number of trips that all buses combined must complete. The task is to determine the minimum amount of time required for all buses to collectively complete at least totalTrips
trips. Keep in mind that it's the combined trips of all buses that we are interested in, not just the trips made by one bus.
Intuition
The solution revolves around the concept of binary search. Initially, we can think that the minimum time required for all buses to complete the required number of trips would be if all buses operated at their fastest possible times, which is why we use the minimum bus time multiplied by the totalTrips
as an upper bound for the time needed.
Given that the buses operate independently, we need to find the point at which the sum of trips made by all buses hits at least totalTrips
. To do this efficiently, we use binary search instead of checking each possible time sequentially.
Binary search works here because as time increases, the number of trips that can be completed is non-decreasing - this is a monotonic relationship, which is a requirement for binary search to be applicable.
The key insight behind the solution is that for a given amount of time t
, we can calculate the total number of trips completed by all buses by summing up t // time[i]
for each bus i
. If this total number of trips is less than totalTrips
, we know we need more time, and if it's more, we have a potential solution but we'll continue to search for a smaller time value that still meets the totalTrips
requirement.
The bisect_left
function from Python's standard library is then used to find the smallest t
where the calculated number of trips is at least totalTrips
. It does this by looking for the leftmost insertion point to maintain sorted order.
Learn more about Binary Search patterns.
Solution Approach
The solution leverages binary search, an efficient algorithm for finding an item from a sorted collection by repeatedly halving the search interval. In this problem, although the starting time
array is not inherently sorted, we are searching through a range of time from 0
to mx
, where mx
is the product of the minimum value in the time
array and totalTrips
. This range of time is conceptually sorted because, as mentioned earlier, the number of total trips completed by all buses increases monotonically as time increases.
The Python bisect_left
function is used in this context to perform the binary search. The bisect_left
function requires a range or a sorted list to search within, a target value to find, and an optional key
function to transform the elements before comparison.
In this problem, our range
represents all possible times from 0
to mx
. The target we are seeking is totalTrips
, our criterion for the minimum trips required.
The key
function is particularly significant here. For every mid-value time x
during the binary search, the key
function calculates the total number of trips that all buses would have completed by that time using the expression sum(x // v for v in time)
where v
is each individual time from the time
list. The //
operator performs integer division, yielding the number of trips a single bus completes in time x
.
The process goes as follows:
- If the calculated total trips for the
mid
time value is less thantotalTrips
, it means more time is needed for the buses to complete the required number of trips, so the search continues to the right half of the current range. - If the total is greater or equal, the
mid
value is a potential solution, but we check to the left half of the range to find if there is a smaller viable time.
This continues until the binary search narrows down to the minimum time where the buses can collectively complete totalTrips
. That is the value returned by the bisect_left
function as a result, hence providing us with the minimum time needed to achieve at least totalTrips
trips by all buses.
Here is the important part of the solution code for reference, focusing on the applied binary search mechanism:
1class Solution:
2 def minimumTime(self, time: List[int], totalTrips: int) -> int:
3 mx = min(time) * totalTrips
4 return bisect_left(
5 range(mx), totalTrips, key=lambda x: sum(x // v for v in time)
6 )
No additional data structures are needed beyond the input time
list and the range object created for the search interval. The binary search pattern itself acts as the algorithmic data structure guiding the search for the correct minimum time.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's walk through a simple example to illustrate the solution approach.
Suppose time = [3, 6, 8]
represents the time taken by three buses to complete a single trip. We need to find the minimum amount of time required for these buses to complete at least totalTrips = 7
trips in total.
First, we determine an upper bound for the binary search based on the fastest bus. The minimum time in time
is 3
, so the maximum conceivable time would be 3 * 7 = 21
. This is because if only the fastest bus were running, it would take 21 units of time to complete 7 trips.
Next, we use binary search to find the minimum time. The range for binary search is from 0
to 21
:
-
In the first iteration, the midpoint (
mid
) of0
and21
is10
. We check how many total trips can be completed by all the buses in10
units of time.- Bus 1 can make
10 // 3 = 3
trips. - Bus 2 can make
10 // 6 = 1
trip. - Bus 3 can make
10 // 8 = 1
trip. - So, the total is
3 + 1 + 1 = 5
trips, which is less than7
. We need more time.
- Bus 1 can make
-
The next
mid
will be the midpoint between11
and21
, which is16
.- Bus 1 can make
16 // 3 = 5
trips. - Bus 2 can make
16 // 6 = 2
trips. - Bus 3 can make
16 // 8 = 2
trips. - Now, the total is
5 + 2 + 2 = 9
trips, which is more than7
. We have a potential solution but we'll continue to search in case there's a lower value that also works.
- Bus 1 can make
-
The next midpoint will be between
11
and15
, giving us13
.- Bus 1 can make
13 // 3 = 4
trips. - Bus 2 can make
13 // 6 = 2
trips. - Bus 3 can make
13 // 8 = 1
trip. - The total is
4 + 2 + 1 = 7
trips, which is exactly what we need.
- Bus 1 can make
-
Even though we've found that
13
minutes would work, we need to make sure there isn't a smaller value that also meets thetotalTrips
requirement, so we look to the left, between11
and12
. -
As we proceed with the search, we find that
12
minutes won't suffice for 7 trips (it only yields 6 trips in total), so we finally conclude that the minimum time needed is13
.
Following the binary search pattern, we get the minimum time required for all buses to collectively complete at least totalTrips
trips which, in this example, is 13
units of time.
Solution Implementation
1from typing import List
2from bisect import bisect_left
3
4class Solution:
5 def minimumTime(self, time: List[int], totalTrips: int) -> int:
6 # Calculate the maximum possible time it would take to complete totalTrips.
7 # This is done by taking the minimum time required for a single trip
8 # and multiplying it by the total number of trips needed.
9 max_time = min(time) * totalTrips
10
11 # Use binary search to find the minimum amount of time needed
12 # to complete the total number of trips.
13 return bisect_left(
14 range(max_time + 1), # The search range includes 0 to max_time
15 totalTrips, # The target value we are searching for
16 # The key function calculates the total number of trips that
17 # can be completed within 'current_time' units of time for each worker.
18 # It implements this by summing up how many trips each worker can complete.
19 key=lambda current_time: sum(current_time // single_time for single_time in time)
20 )
21
1class Solution {
2 /**
3 * Finds the minimum time required to complete the total number of trips.
4 *
5 * @param time An array where each element represents the time it takes a bus to make one trip.
6 * @param totalTrips The total number of trips that need to be completed.
7 * @return The minimum time required to complete the total number of trips.
8 */
9 public long minimumTime(int[] time, int totalTrips) {
10 // Initialize the variable to store the minimum time needed for one trip.
11 int minTime = time[0];
12
13 // Find the smallest trip time among all buses.
14 for (int tripTime : time) {
15 minTime = Math.min(minTime, tripTime);
16 }
17
18 // Set initial left and right bounds for binary search.
19 long left = 1;
20 long right = (long) minTime * totalTrips;
21
22 // Perform binary search to find the minimum time required.
23 while (left < right) {
24 long mid = (left + right) >> 1; // Calculate the midpoint for the current range.
25 long count = 0; // Initialize the count of trips that can be made by all buses in 'mid' time.
26
27 // Calculate how many total trips can be made by all buses in 'mid' time.
28 for (int tripTime : time) {
29 count += mid / tripTime;
30 }
31
32 // If the count of trips is equal to or higher than required, search in the left half.
33 if (count >= totalTrips) {
34 right = mid;
35 // Else search in the right half.
36 } else {
37 left = mid + 1;
38 }
39 }
40
41 // The left pointer will point to the minimum time required when the search completes.
42 return left;
43 }
44}
45
1#include <vector>
2#include <algorithm>
3
4class Solution {
5public:
6 // Function to find the minimum time needed to make a number of trips.
7 long long minimumTime(vector<int>& timePerTrip, int totalTrips) {
8 // Find the minimum time per trip to estimate lower bound
9 int minTime = *min_element(timePerTrip.begin(), timePerTrip.end());
10 // Lower bound starts at 1 (minimum possible time)
11 // Upper bound is the product of the minimum time per trip and the total number of trips
12 long long left = 1;
13 long long right = static_cast<long long>(minTime) * totalTrips;
14
15 // Perform binary search to find the minimum time needed
16 while (left < right) {
17 // Calculating the midpoint of our search interval
18 long long mid = (left + right) / 2;
19 long long tripsCompleted = 0;
20
21 // Counting the total number of trips that can be completed in 'mid' time
22 for (int time : timePerTrip) {
23 tripsCompleted += mid / time;
24 }
25
26 // If the number of completed trips is at least the required amount,
27 // we can try to find a smaller maximum time
28 if (tripsCompleted >= totalTrips) {
29 right = mid;
30 // If not, we must increase the minimum time
31 } else {
32 left = mid + 1;
33 }
34 }
35 // When the loop exits, 'left' will be our answer because 'right' will have converged to 'left'
36 return left;
37 }
38};
39
1function minimumTime(timePerTrip: number[], totalTrips: number): number {
2 // Calculate the minimum time for a single trip to set the lower limit for binary search
3 const minTime = Math.min(...timePerTrip);
4 // Set initial lower bound as 1 since time cannot be 0
5 let left: number = 1;
6 // Set upper bound as minTime * totalTrips, assuming all trips at slowest speed
7 let right: number = minTime * totalTrips;
8
9 // Perform binary search to find the minimum time needed
10 while (left < right) {
11 // Find the mid-point time
12 const mid: number = Math.floor((left + right) / 2);
13 let tripsCompleted: number = 0;
14
15 // Compute the number of trips that can be completed within 'mid' time
16 for (const time of timePerTrip) {
17 tripsCompleted += Math.floor(mid / time);
18 }
19
20 // If the calculated trips are equal or more than required, try to find a smaller time window
21 if (tripsCompleted >= totalTrips) {
22 right = mid;
23 } else {
24 // Otherwise, increase left to find a time period that allows completing the totalTrips
25 left = mid + 1;
26 }
27 }
28
29 // When the binary search is complete, 'left' will contain the minimum time needed
30 return left;
31}
32
Time and Space Complexity
Time Complexity
The given code uses a binary search approach to find the minimum time required to make totalTrips
using a list of time
where each time[i]
represents the time needed to complete one trip.
The time complexity of the binary search is O(log(max_time))
, where max_time = min(time) * totalTrips
. This represents the maximum amount of time it might take if the slowest worker alone had to complete all totalTrips
.
Within each step of the binary search, the code executes a sum operation that runs in O(n)
time, where n
is the number of workers (or the length of the time
list). This sum operation calculates the total number of trips completed within a given time frame.
Therefore, the total time complexity of the code is O(n * log(max_time))
.
Space Complexity
The space complexity of the code is O(1)
since it only uses constant extra space for variables such as mx
and the lambda function. There is no use of any additional data structures that grow with the input size.
Learn more about how to find time and space complexity quickly using problem constraints.
What's the relationship between a tree and a graph?
Recommended Readings
Binary Search Speedrun For each of the Speedrun questions you will be given a binary search related problem and a corresponding multiple choice question The multiple choice questions are related to the techniques and template s introduced in the binary search section It's recommended that you have gone through at
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
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.