1229. Meeting Scheduler
Problem Description
The problem is about finding a mutual meeting time slot between two people given their individual schedules and a required meeting duration. Each person's schedule is represented by a list of non-overlapping time slots where a time slot is an array [start, end]
showing availability from start
to end
. The goal is to find the earliest starting time slot that is available in both schedules and lasts at least for the given duration
. If there's no such common time slot, we return an empty array.
Intuition
To solve this problem, we can use the two-pointer technique. Since no individual time slots overlap within a single person's schedule, we can sort both schedules by the starting times of the slots. We then compare the slots from both schedules to find overlapping slots.
We use two pointers i
and j
to traverse slots1
and slots2
respectively. In each iteration, we find the latest start time by taking the maximum of slots1[i][0]
and slots2[j][0]
and the earliest end time by taking the minimum of slots1[i][1]
and slots2[j][1]
. If the overlapping time between the latest start and earliest end is greater than or equal to the required duration
, we have found a suitable time slot and return the start and end of this meeting slot.
If the overlap is not sufficient, we move the pointer forward in the list which has the earlier ending time slot, hoping to find a longer slot that might overlap with the other person's next slot. This process continues until we either find a suitable slot or exhaust all available slots in either list.
Learn more about Two Pointers and Sorting patterns.
Solution Approach
The provided Python solution follows a straightforward two-pointer approach to solve the problem:
-
Sorting the time slots: Both
slots1
andslots2
are sorted based on their starting times. This ensures that we are always considering the earliest available slots first and eliminates the need for checking past time slots. Sorting is crucial as it sets up the structure for the two-pointer technique to work effectively. -
Two-pointer Technique: Two pointers,
i
andj
, are used to iterate throughslots1
andslots2
respectively. At each step,i
refers to the current slot in the first person's schedule, andj
refers to the current slot in the second person's schedule. -
Finding Overlaps: For the current pair of time slots pointed by
i
andj
, we calculate the overlap by determining the maximum of the two start times and the minimum of the two end times. The variablesstart
andend
are used to record these values:1start = max(slots1[i][0], slots2[j][0]) 2end = min(slots1[i][1], slots2[j][1])
-
Checking Duration: We then check if the overlapping duration is greater than or equal to the required
duration
by subtractingstart
fromend
:1if end - start >= duration:
If the condition is met, we have found a valid time slot and can return
[start, start + duration]
as the solution. -
Advancing the Pointers: If the overlapping time slot is not long enough, we need to discard the time slot that ends earlier and move forward. This decision is made by comparing the end times of the current time slots pointed by
i
andj
. The pointer corresponding to the slot with the earlier end time is incremented:1if slots1[i][1] < slots2[j][1]: 2 i += 1 3else: 4 j += 1
This step ensures that we're always trying to find overlap with the nearest possible future slot.
-
Continuation and Termination: The above steps are continued in a loop until one of the lists is exhausted (i.e.,
i
reaches the end ofslots1
orj
reaches the end ofslots2
). If no common time slots with sufficient duration are found by the end of either list, we return an empty array[]
as specified.
This algorithm makes efficient use of the sorted structure of the time slots and the two-pointer technique to minimize the number of comparisons and quickly find the first suitable time slot, achieving a time complexity that is linear in the size of the input time slots after the initial sort.
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 consider an example to illustrate the solution approach:
- Person A's schedule (
slots1
):[[10, 50], [60, 120], [140, 210]]
- Person B's schedule (
slots2
):[[0, 15], [25, 50], [60, 70], [80, 100]]
- Required duration for the meeting:
8 minutes
-
Sorting the time slots: Both schedules are already sorted based on their starting times, eliminating the need for a sort operation in this example.
-
Two-pointer Technique: Initialize two pointers,
i
forslots1
andj
forslots2
. Initially,i = 0
andj = 0
. -
Finding Overlaps:
-
First Comparison:
slots1[i] = [10, 50]
andslots2[j] = [0, 15]
- The overlap is from
10
(max of 10 and 0) to15
(min of 50 and 15), which is5
minutes long. This is not enough for the8-minute
duration. - Move the pointer
j
forward becauseslots2[j][1]
is less thanslots1[i][1]
.
-
Second Comparison:
slots1[i] = [10, 50]
andslots2[j] = [25, 50]
- The overlap is from
25
to50
, which is25
minutes long and suffices for the8-minute
duration. - We have found a suitable slot, so we can return the result
[25, 25 + 8]
, which is[25, 33]
.
-
This result indicates that Person A and Person B can successfully schedule a meeting starting at 25
minutes past the hour and lasting until 33
minutes past the hour.
The method described above is efficient because it continuously seeks the earliest possible meeting time by looking for overlaps in the schedules and moves forward based on the end times of the current slots. As soon as a fitting time slot is found, it returns the result without unnecessary comparisons of later time slots, saving time and computation.
Solution Implementation
1from typing import List
2
3class Solution:
4 def minAvailableDuration(self, slots1: List[List[int]], slots2: List[List[int]], duration: int) -> List[int]:
5 # Sort the time slots for both people to allow for easy comparison
6 slots1.sort()
7 slots2.sort()
8
9 # Initialize variables to track current index in slots1 and slots2
10 index1 = index2 = 0
11
12 # Get the total number of slots for both people
13 total_slots1 = len(slots1)
14 total_slots2 = len(slots2)
15
16 # Iterate over the slots until at least one person's slots are fully checked
17 while index1 < total_slots1 and index2 < total_slots2:
18 # Find the overlapping start time of the current slots
19 overlap_start = max(slots1[index1][0], slots2[index2][0])
20 # Find the overlapping end time of the current slots
21 overlap_end = min(slots1[index1][1], slots2[index2][1])
22
23 # If the overlapping period is greater than or equal to the required duration
24 if overlap_end - overlap_start >= duration:
25 # Return the start time and start time plus the duration
26 return [overlap_start, overlap_start + duration]
27
28 # If the end time in slots1 is before the end time in slots2,
29 # increase the index for slots1 to check the next slot
30 if slots1[index1][1] < slots2[index2][1]:
31 index1 += 1
32 else:
33 # Otherwise, increase the index for slots2 to check the next slot
34 index2 += 1
35
36 # If no overlapping period long enough for the meeting is found, return an empty list
37 return []
38
39# Example usage:
40# sol = Solution()
41# result = sol.minAvailableDuration([[10, 50], [60, 120], [140, 210]], [[0, 15], [60, 70]], 8)
42# print(result) # This will output: [60, 68]
43
1class Solution {
2 public List<Integer> minAvailableDuration(int[][] slots1, int[][] slots2, int duration) {
3 // Sort the time slots for both people based on the start times
4 Arrays.sort(slots1, (a, b) -> a[0] - b[0]);
5 Arrays.sort(slots2, (a, b) -> a[0] - b[0]);
6
7 int index1 = 0; // Index for navigating person 1's time slots
8 int index2 = 0; // Index for navigating person 2's time slots
9 int len1 = slots1.length; // Total number of slots for person 1
10 int len2 = slots2.length; // Total number of slots for person 2
11
12 // Iterate through both sets of slots
13 while (index1 < len1 && index2 < len2) {
14 // Calculate the overlap start time
15 int overlapStart = Math.max(slots1[index1][0], slots2[index2][0]);
16 // Calculate the overlap end time
17 int overlapEnd = Math.min(slots1[index1][1], slots2[index2][1]);
18
19 // Check if the overlapping duration is at least the required duration
20 if (overlapEnd - overlapStart >= duration) {
21 // If so, return the start time of the meeting and start time plus duration
22 return Arrays.asList(overlapStart, overlapStart + duration);
23 }
24
25 // Move to the next slot in the list that has an earlier end time
26 if (slots1[index1][1] < slots2[index2][1]) {
27 index1++;
28 } else {
29 index2++;
30 }
31 }
32
33 // If no common slot is found that fits the duration, return an empty list
34 return Collections.emptyList();
35 }
36}
37
1#include <vector>
2#include <algorithm>
3
4class Solution {
5public:
6 std::vector<int> minAvailableDuration(std::vector<std::vector<int>>& slots1, std::vector<std::vector<int>>& slots2, int duration) {
7 // Sort both sets of time slots in ascending order based on the start times
8 std::sort(slots1.begin(), slots1.end());
9 std::sort(slots2.begin(), slots2.end());
10
11 // Use indices to track the current slot in each set
12 int index1 = 0;
13 int index2 = 0;
14
15 // Loop through both sets of time slots to find a common free duration
16 while (index1 < slots1.size() && index2 < slots2.size()) {
17 // Find the latest start time and earliest end time between two slots
18 int latestStart = std::max(slots1[index1][0], slots2[index2][0]);
19 int earliestEnd = std::min(slots1[index1][1], slots2[index2][1]);
20
21 // Check if the overlap duration between two slots is greater than or equal to the desired duration
22 if (earliestEnd - latestStart >= duration) {
23 // If so, return the start time and start time plus duration as the available duration
24 return {latestStart, latestStart + duration};
25 }
26
27 // Move to the next slot in the set with the earlier ending time
28 if (slots1[index1][1] < slots2[index2][1]) {
29 ++index1;
30 } else {
31 ++index2;
32 }
33 }
34
35 // If no common free duration is found, return an empty vector
36 return {};
37 }
38};
39
1function minAvailableDuration(slots1: number[][], slots2: number[][], duration: number): number[] {
2 // Sort both sets of time slots by start time in ascending order
3 slots1.sort((a, b) => a[0] - b[0]);
4 slots2.sort((a, b) => a[0] - b[0]);
5
6 // Initialize pointers to traverse the slots
7 let index1 = 0;
8 let index2 = 0;
9
10 while (index1 < slots1.length && index2 < slots2.length) {
11 // Calculate the maximum start time and the minimum end time between two slots
12 const latestStart = Math.max(slots1[index1][0], slots2[index2][0]);
13 const earliestEnd = Math.min(slots1[index1][1], slots2[index2][1]);
14
15 // Check if there is a slot sufficient for the required duration
16 if (earliestEnd - latestStart >= duration) {
17 // If found, return the time range starting from latestStart and continuing for the duration
18 return [latestStart, latestStart + duration];
19 }
20
21 // Move to the next slot in the list with the earlier end time
22 if (slots1[index1][1] < slots2[index2][1]) {
23 index1++; // Move the pointer in the first list forward
24 } else {
25 index2++; // Move the pointer in the second list forward
26 }
27 }
28
29 // If no matching slots were found, return an empty array
30 return [];
31}
32
33// Example usage:
34// const result = minAvailableDuration([[10, 50], [60, 120], [140, 210]], [[0, 15], [60, 70]], 8);
35// console.log(result); // Should log a valid time slot with at least 8 minutes of duration, or an empty array if there's no slot.
36
Time and Space Complexity
Time Complexity
The given algorithm primarily consists of two parts: sorting the time slots and then iterating through the sorted lists to find a common available duration.
First, we sort both slots1
and slots2
. Assuming that slots1
has m
intervals and slots2
has n
intervals, the time taken to sort these lists using a comparison-based sorting algorithm, like Timsort (Python's default sorting algorithm), is O(m log m)
for slots1
and O(n log n)
for slots2
.
After sorting, we have a while loop that runs until we reach the end of one of the slot lists. In the worst case, we'll compare each pair of slots once, which leads to a complexity of O(m + n)
since each list is traversed at most once.
The overall time complexity is the sum of the complexities of sorting and iterating through the slot lists, which is O(m log m + n log n + m + n)
. However, the log
factor dominates the linear factor in computational complexity analysis for large values. Hence, the time complexity simplifies to O(m log m + n log n)
.
Space Complexity
The space complexity of the algorithm is determined by the space we use to sort slots1
and slots2
. Since the Timsort algorithm can require a certain amount of space for its operation, the space complexity is O(m+n)
due to the auxiliary space used in sorting. All other operations use a constant amount of space, and no additional space is used that depends on the input size, so the space complexity remains O(m+n)
.
Learn more about how to find time and space complexity quickly using problem constraints.
Which of the tree traversal order can be used to obtain elements in a binary search tree in sorted order?
Recommended Readings
Tech Interview Pattern Two Pointers Introduction If you prefer videos here's a super quick introduction to Two Pointers div class responsive iframe iframe src https www youtube com embed xZ4AfXHQ1VQ title YouTube video player frameborder 0 allow accelerometer autoplay clipboard write encrypted media gyroscope picture in picture allowfullscreen iframe div Two pointers is a common interview
Sorting Summary Comparisons We presented quite a few sorting algorithms and it is essential to know the advantages and disadvantages of each one The basic algorithms are easy to visualize and easy to learn for beginner programmers because of their simplicity As such they will suffice if you don't know any advanced
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
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.