Problem
You are given a positive integer days
representing the total number of days an employee is available for work (starting from day 1). You are also given a 2D array meetings
of size n
where, meetings[i] = [start_i, end_i]
represents the starting and ending days of meeting i
(inclusive).
Return the count of days when the employee is available for work but no meetings are scheduled.
Note: The meetings may overlap.
Examples
Example 1:
|
|
Example 2:
|
|
Example 3:
|
|
Constraints:
1 <= days <= 10^9
1 <= meetings.length <= 10^5
meetings[i].length == 2
1 <= meetings[i][0] <= meetings[i][1] <= days
Solution
Method 1 - Using the set
We can
- Use a set (or boolean array) to track days that are occupied by meetings.
- Traverse through the list of meetings, and mark all days within each meeting’s range (
start_i
toend_i
) as occupied in the tracking set/array. - Any day that is not marked as occupied after processing all meetings is considered a free day.
Steps
- Initialise a data structure (like a set or an array) to track which days are taken by meetings.
- Iterate through the
meetings
list, marking the occupied days. - Finally, count the days between 1 and
days
that are not occupied.
This approach efficiently handles overlapping meetings because duplicates are eliminated in the set or overwritten in the boolean array. But will result in TLE for large days.
Code
|
|
|
|
Complexity
- ⏰ Time complexity:
O(n * m + days)
, wheren
is the number of meetings andm
is the average range of meeting days.- For creating the list of occupied days:
O(n * m)
- Counting free days:
O(days)
- For creating the list of occupied days:
- 🧺 Space complexity:
O(days)
Method 2 - Lazy marking using difference array
This method uses a Difference Array to efficiently mark ranges of occupied days without explicitly iterating over every day in each range. Instead of directly marking all days in a range as occupied, the approach works by marking the start and end points of ranges using a difference array. The occupancy status for all days is then computed using prefix sums over the difference array.
Steps
- Difference Array Setup:
- Create a difference array (
difference[]
) to represent changes at range boundaries:- Increment
difference[start]
by1
to mark the start of a range. - Decrement
difference[end + 1]
by1
to mark the end of a range.
- Increment
- This efficiently marks the ranges without manually iterating over all individual days within them.
- Create a difference array (
- Occupancy Array Construction:
- Use a prefix sum across the difference array to construct the occupancy status of each day:
- If the sum is positive at a day, it is occupied.
- Otherwise, it is free.
- Use a prefix sum across the difference array to construct the occupancy status of each day:
- Count Free Days:
- Iterate over the occupancy array and count days that are not occupied.
Code
|
|
|
|
Complexity
- ⏰ Time complexity:
O(n + days)
, wheren
is the number of meetings.- Updating difference array:
O(n)
→ Loop through alln
meetings. - Constructing occupancy array:
O(days)
→ Loop through alldays
. - Counting free days:
O(days)
→ Loop through alldays
.
- Updating difference array:
- 🧺 Space complexity:
O(days)
- Difference array:
O(days)
. - Occupancy array:
O(days)
.
- Difference array:
Dry Run
Suppose:
days = 10
meetings = [[1, 3], [5, 7]]
Execution:
Difference Array:
1
[0, +1, 0, 0, -1, +1, 0, 0, -1, 0, 0]
Prefix Sum (
occupancy status
):1
[False, True, True, True, False, True, True, True, False, False, False]
Free Days:
- Days
{4, 8, 9, 10}
are free. - Result:
4
.
- Days
Method 3 - Using Sorting and Interval Merging
Instead of marking specific days, we can process the meetings as intervals, merge overlapping intervals, and calculate the total occupied range. Using this strategy, we minimise the computation to just the required number of intervals instead of days
.
Steps
- Sort Intervals:
- Sort the meetings based on their start times. This helps us merge overlapping or adjacent intervals efficiently.
- Merge Intervals:
- Traverse the sorted intervals, merging overlapping ones into a single interval.
- Count Occupied Days:
- Calculate the total number of days occupied by these merged intervals.
- Subtract from Total Days:
- The total available days (
days
) minus the occupied days gives the count of free days.
- The total available days (
Code
|
|
|
|
Complexity
- ⏰ Time complexity:
O(n log n)
, wheren
is the number of meetings.- Sorting intervals:
O(n log n)
wheren
is the number of meetings. - Merging intervals:
O(n)
since we iterate through the list ofmeetings
.
- Sorting intervals:
- 🧺 Space complexity:
O(1)
auxiliary space apart from the input.
Dry Run
Suppose:
days = 10
meetings = [[1, 3], [5, 7], [2, 6]]
Execution:
Sorted Meetings:
1
[[1, 3], [2, 6], [5, 7]]
Merge Process:
- First interval:
[1, 3]
- Extend to
[1, 6]
(merging[2, 6]
) - Extend to
[1, 7]
(merging[5, 7]
).
- First interval:
Total Occupied Days:
- Length of merged interval
[1, 7] = 7 days
.
- Length of merged interval
Free Days:
1
10 - 7 = 3
Result: 3
free days.