849. Maximize Distance to Closest Person - Detailed Explanation
Problem Statement
You are given an array of integers, seats
, where:
seats[i] = 1
indicates that the i-th seat is occupied.seats[i] = 0
indicates that the i-th seat is empty.
You want to choose an empty seat such that the distance to the closest person is maximized. Return the maximum distance to the closest person you can achieve by choosing an optimal seat.
Example 1
- Input:
[1,0,0,0,1,0,1]
- Output:
2
- Explanation:
The best place to sit is at index 2 (or index 3). If you choose index 2, the closest occupied seats are at index 0 and index 4. The distances are 2 and 2 respectively, so the minimum distance is 2.
Example 2
- Input:
[1,0,0,0]
- Output:
3
- Explanation:
The row starts with an occupied seat at index 0 and then three empty seats. Sitting at the last seat (index 3) gives a distance of 3 from the occupied seat at index 0.
Example 3
- Input:
[0,1]
- Output:
1
- Explanation:
The only empty seat is at index 0, and the nearest person is at index 1, so the maximum distance is 1.
Constraints
2 <= seats.length <= 20,000
seats[i]
is either0
or1
.- There is at least one empty seat and at least one occupied seat.
Hints to Guide Your Approach
-
Consider the Gaps:
Look at the segments (or gaps) of consecutive empty seats. The maximum distance you can achieve is determined by:- The gap at the beginning of the array (if the row starts with empty seats).
- The gap at the end of the array (if the row ends with empty seats).
- The gaps in between two occupied seats (for which you’d want to sit in the middle).
-
One-Pass Strategy:
As you iterate through the array, track the index of the last occupied seat. Use this information to compute the distance for the gap between the current occupied seat and the previous one. Don’t forget to check the gap at the very end of the row after your iteration.
Approaches to Solve the Problem
Approach 1: Brute Force
Explanation
-
Idea:
For each empty seat, determine the distance to the closest occupied seat by scanning to the left and right. -
Steps:
- For every seat that is empty (
0
), scan leftward until you find an occupied seat. - Similarly, scan rightward until you find an occupied seat.
- The effective distance for that seat is the minimum of the two distances.
- Track the maximum among all empty seats.
- For every seat that is empty (
-
Complexity Analysis:
- Time Complexity: O(n²) in the worst-case scenario (if many seats are empty).
- Space Complexity: O(1)
-
Pitfalls:
This approach might be too slow when the number of seats is large (up to 20,000).
Approach 2: One-Pass Optimal Approach
Explanation
-
Idea:
Instead of scanning from each empty seat, use a single pass through the array to calculate the maximum gap. -
Steps:
- Initialize:
- Set a pointer
prev
to track the index of the last occupied seat (start withprev = -1
). - Initialize
max_distance
to 0.
- Set a pointer
- Iterate through the array:
- When you encounter an occupied seat:
- Left Gap: If
prev == -1
, it means this is the first occupied seat. The maximum distance for seats before it is the index value (i.e.,i
). - Middle Gap: Otherwise, the maximum distance you can get in the gap between
prev
and the current occupied seat is half of the gap:(i - prev) // 2
.
- Left Gap: If
- Update
prev
to the current index.
- When you encounter an occupied seat:
- Right Gap:
- After the loop, check the gap from the last occupied seat to the end of the array (
(n - 1 - prev)
), and updatemax_distance
accordingly.
- After the loop, check the gap from the last occupied seat to the end of the array (
- Initialize:
-
Complexity Analysis:
- Time Complexity: O(n) – a single pass through the array.
- Space Complexity: O(1)
Visual Example
Consider the array: [1,0,0,0,1,0,1]
-
Step 1:
prev = -1
,max_distance = 0
-
Iteration:
-
At index 0:
seats[0] = 1
- Since
prev = -1
, updatemax_distance = 0
(distance from the start is 0). - Set
prev = 0
.
- Since
-
At indices 1, 2, 3:
seats[1,2,3] = 0
(skip them until the next occupied seat). -
At index 4:
seats[4] = 1
- Calculate gap:
i - prev = 4 - 0 = 4
- Distance if sitting in the middle:
4 // 2 = 2
- Update
max_distance = max(0, 2) = 2
- Set
prev = 4
.
- Calculate gap:
-
At index 5:
seats[5] = 0
(skip). -
At index 6:
seats[6] = 1
- Calculate gap:
6 - 4 = 2
- Distance in the middle:
2 // 2 = 1
max_distance
remains 2.- Set
prev = 6
.
- Calculate gap:
-
-
After the Loop:
Check gap from index 6 to end:n - 1 - prev = 6 - 6 = 0
.
So, the final answer is 2.
Code Implementations
Python Code
Below is the Python implementation using the optimal one-pass approach. A brute force version can be implemented similarly, but it is not efficient for larger inputs.
Java Code
Below is the Java implementation using the optimal one-pass approach.
Step-by-Step Walkthrough & Visual Examples
Walkthrough of the Optimal Approach
-
Initialization:
- Set
prev = -1
to indicate that no occupied seat has been seen yet. - Set
max_distance = 0
.
- Set
-
Iterate through the array:
- For the first occupied seat:
- If it is found at index
i
andprev == -1
, updatemax_distance = i
(because you can sit at index 0 and have a distance ofi
).
- If it is found at index
- For subsequent occupied seats:
- When you find an occupied seat at index
i
, the gap betweenprev
(last occupied) andi
isi - prev
. The maximum distance achievable in that gap is(i - prev) // 2
(by sitting in the middle). - Update
max_distance
if this value is larger than the current value.
- When you find an occupied seat at index
- Update
prev
to the current indexi
.
- For the first occupied seat:
-
After the loop:
- Check the gap from the last occupied seat to the end of the row. This gap gives a potential distance of
(n - 1 - prev)
. - Update
max_distance
if needed.
- Check the gap from the last occupied seat to the end of the row. This gap gives a potential distance of
-
Return:
- The final
max_distance
is the answer.
- The final
Common Mistakes
- Ignoring Edge Gaps:
Failing to correctly handle the gap at the beginning (if the first seat is empty) or at the end (if the last seat is empty) can lead to wrong answers. - Incorrect Division for Middle Gaps:
Using the full gap length instead of half (because you always want to sit in the middle of a gap) will result in an overestimate. - Not Updating
prev
:
Forgetting to update theprev
index after processing an occupied seat may cause incorrect gap calculations.
Edge Cases
- All Empty Except One:
If there is only one occupied seat (at either end), the answer is the distance from that seat to the far end of the row. - Only Two Seats:
With only two seats and one occupied, the answer is always 1.
Alternative Variations & Related Problems
- Variations:
- Maximizing distance in a circular seating arrangement.
- Minimizing the maximum distance if you want to make everyone as close as possible.
- Related Problems for Further Practice:
- "Exam Room" – where you assign seats to maximize distance.
- "Capacity To Ship Packages Within D Days" – involves binary search on the answer.
- "Split Array Largest Sum" – involves dividing an array optimally.
Complexity Recap
- Brute Force Approach:
- Time: O(n²)
- Space: O(1)
- Optimal One-Pass Approach:
- Time: O(n)
- Space: O(1)
Both methods are viable given the problem constraints, but the optimal approach is much more efficient and preferred for larger inputs.
GET YOUR FREE
Coding Questions Catalog
