1040. Moving Stones Until Consecutive II
Problem Description
The problem presents a game scenario with stones placed at various positions on the X-axis. You are given an array, stones
, representing the positions of the stones. A stone becomes an endpoint stone if it is at either end of the sequence (smallest or largest position). The objective is to make moves to remove a stone from being an endpoint stone by placing it at a new unoccupied position.
The rules of the game state that you cannot move an endpoint stone to a position where it continues to be an endpoint stone. The game ends when you cannot make any more moves and the stones are in three consecutive positions.
The problem asks for two outputs:
- The minimum number of moves required, represented by
answer[0]
. - The maximum number of moves that can be made, represented by
answer[1]
.
Intuition
To figure out the solution, understand the following key points:
Minimum Moves: To compute the minimum number of moves (mi
), we aim to bring the stones into a consecutive sequence with as few moves as possible. This usually entails moving endpoint stones closer to the center. The edge case is when there's exactly one gap between two stones that's equal to the number of stones minus two - this would require two moves to fill in the positions and make the sequence consecutive.
Maximum Moves: For the maximum number of moves (mx
), think of moving the endpoint stones away from the center to the farthest position possible inside the current stone range, but one at a time, to ensure they are not considered an endpoint stone anymore. It's a game of maximizing the number of moves by utilizing the current gaps to the fullest.
The code's approach to solving for mi
and mx
involves sorting the stones array to process stone positions in a defined order. It then utilizes sliding window technique to find the minimum moves required to create a consecutive sequence. The maximum number of moves are computed considering the largest gaps on either end of the sorted array.
Learn more about Math, Two Pointers and Sorting patterns.
Solution Approach
The solution follows these steps:
-
Sort the Array: The first step requires sorting the
stones
array. Sorting helps identify the endpoints easily and arrange stones in an ascending order which is crucial for the next steps. -
Initialize Variables: A variable
n
is used to store the number of stones, andmi
is initialized with the value ofn
, to be later updated with the actual minimum moves. -
Compute the Maximum Moves (
mx
): Calculate the maximum moves which can be made by considering one of two scenarios, whichever yields a larger value:- Moving the stone from the start to just after the second stone, i.e.,
stones[-1] - stones[1] + 1
minus the number of stones excluding the moved stone. - Moving the second last stone to just before the first stone, i.e.,
stones[-2] - stones[0] + 1
minus the number of stones excluding the moved stone.
The
+1
in the formulas accounts for the fact that we want to include both endpoints in the range. These operations represent the case where you move the first or last stone to get the most number of moves without being an endpoint stone in the next move. - Moving the stone from the start to just after the second stone, i.e.,
-
Sliding Window for Minimum Moves (
mi
): The next part of the code uses a sliding window approach to calculate the minimum moves. Two pointersi
(start of the window) andj
(end of the window) iterate over the array to determine the smallest movement of stones necessary to reach a position where stones are in a consecutive sequence.- The window size cannot be greater than
n
, so if the current window size exceedsn
, the start of the window (i
) is moved up. - If the end of the window minus the start of the window plus one (
j - i + 1
) equalsn - 1
and the range (x
-stones[i]
) isn - 2
, this means there is exactly one gap that requires two moves to fill. Otherwise, the minimum moves (mi
) are updated ton
minus the size of the current window (n - (j - i + 1)
).
- The window size cannot be greater than
The code uses these techniques and conditional checks to ensure we find the least and the maximum number of moves to reach the game's end goal.
Finally, the function returns the list [mi, mx]
containing the minimum and maximum moves possible.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Assuming we have an array of stone positions given as stones = [1, 3, 5, 7]
. Let's apply the solution approach to this array to understand how we get our answer
with the minimum and maximum number of moves.
-
Sort the Array: The initial array is already sorted, which looks like
[1, 3, 5, 7]
. The endpoints here are1
and7
. -
Initialize Variables: The number of stones
n
is4
in this case. We'll start withmi = 4
for the minimum moves. -
Compute the Maximum Moves (
mx
):- Moving the stone from the start (
1
) to just after the second stone (3
) would be:7 - 3 + 1 = 5
minus the number of stones excluding the moved one, which is3
, so5 - 3 = 2
. - Moving the second last stone (
5
) to just before the first stone (1
) would yield:5 - 1 + 1 = 5
minus the remaining stones, again3
, resulting in2
.
So, in either case, the
mx
is2
. - Moving the stone from the start (
-
Sliding Window for Minimum Moves (
mi
):- Using a sliding window, we check consecutive positions in the sorted array to calculate the fewest moves to get stones in a consecutive sequence.
- We start with the window
i = 0
and expandj
till we reach2
, so we look at[1, 3, 5]
. The window size here is3
, which is one less thann
, and the range is5 - 1 = 4
, which is notn - 2
. The stones in this window already form a consecutive sequence, so no moves needed. - Now let's consider the window
[3, 5, 7]
withi = 1
andj = 3
. Similar to the first window, stones are already consecutive, and no moves are required. - Since the stones within these windows are consecutive and we never encounter a situation with a gap of
n - 2
,mi
is0
, as we don't need any moves to achieve the consecutive sequence.
Finally, we find that the minimum number of moves, mi
, is 0
and the maximum number of moves, mx
, is 2
. Hence, answer = [mi, mx]
would return [0, 2]
.
Solution Implementation
1class Solution:
2 def numMovesStonesII(self, stones: List[int]) -> List[int]:
3 stones.sort() # First, sort the stones in non-decreasing order.
4 total_stones = len(stones) # Find the total number of stones.
5
6 # Initialize the minimum and maximum moves.
7 min_moves = total_stones
8 max_moves = max(stones[-1] - stones[1] - total_stones + 2,
9 stones[-2] - stones[0] - total_stones + 2)
10
11 # 'i' will denote the start index of a window of stones
12 i = 0
13 for j, current_stone in enumerate(stones):
14 # Slide the window towards the right if it's too long,
15 # i.e., the length of the window is greater than the number of stones
16 while current_stone - stones[i] + 1 > total_stones:
17 i += 1
18
19 # Check for a special case where there's exactly one spot open
20 # within a group of consecutively placed stones - in this case,
21 # two moves are always required.
22 if j - i + 1 == total_stones - 1 and current_stone - stones[i] == total_stones - 2:
23 min_moves = min(min_moves, 2)
24 else:
25 # Otherwise, the minimum moves are the total stones minus the size of the current window
26 min_moves = min(min_moves, total_stones - (j - i + 1))
27
28 # Return the results as a list containing the minimum and maximum moves.
29 return [min_moves, max_moves]
30
1class Solution {
2 public int[] numMovesStonesII(int[] stones) {
3 // Sort the stone positions
4 Arrays.sort(stones);
5
6 // Number of stones
7 int numStones = stones.length;
8
9 // Minimum moves initialized to the total number of stones
10 int minMoves = numStones;
11
12 // The maximum moves are calculated using the end positions of the stones minus their starting positions,
13 // then subtracting the total number of stones minus one from it.
14 int maxMoves = Math.max(stones[numStones - 1] - stones[1], stones[numStones - 2] - stones[0]) + 1 - numStones;
15
16 // Initialize two pointers for the sliding window technique
17 for (int i = 0, j = 0; j < numStones; ++j) {
18 // While the difference between stones at j and i pointers plus one is greater than the total number of stones,
19 // increment the i pointer.
20 while (stones[j] - stones[i] + 1 > numStones) {
21 ++i;
22 }
23
24 // Check if there's a gap for one stone to make two moves
25 if (j - i + 1 == numStones - 1 && stones[j] - stones[i] == numStones - 2) {
26 minMoves = Math.min(minMoves, 2);
27 } else {
28 // Otherwise, calculate the minimum moves as the difference between total stones and the number of stones
29 // within the current range.
30 minMoves = Math.min(minMoves, numStones - (j - i + 1));
31 }
32 }
33
34 // Return the minimum and maximum moves as an array
35 return new int[]{minMoves, maxMoves};
36 }
37}
38
1class Solution {
2public:
3 vector<int> numMovesStonesII(vector<int>& stones) {
4 // first, sort the positions of stones
5 sort(stones.begin(), stones.end());
6
7 // n represents the total number of stones
8 int numStones = stones.size();
9
10 // initialize minimum moves to the number of stones, which will be updated later
11 int minMoves = numStones;
12
13 // the maximum moves are calculated by considering the empty spaces at the ends of the sequence
14 int maxMoves = max(stones[numStones - 1] - stones[1], stones[numStones - 2] - stones[0]) - (numStones - 2);
15
16 // using two-pointer technique to find the smallest window that can contain all the stones
17 for (int start = 0, end = 0; end < numStones; ++end) {
18 // Slide window: if the current window cannot fit in a consecutive sequence of size numStones
19 while (stones[end] - stones[start] + 1 > numStones) {
20 ++start; // increment the start of the window
21 }
22
23 // checking for the case with n - 1 consecutive stones with one space in between them
24 if (end - start + 1 == numStones - 1 && stones[end] - stones[start] == numStones - 2) {
25 // 2 moves are required in this specific case
26 minMoves = min(minMoves, 2);
27 } else {
28 // otherwise, the minimum moves are determined by the size of the current window
29 minMoves = min(minMoves, numStones - (end - start + 1));
30 }
31 }
32
33 // return the pair of minimum and maximum moves
34 return {minMoves, maxMoves};
35 }
36};
37
1function numMovesStonesII(stones: number[]): number[] {
2 // Sort stones array so stones are in non-decreasing order
3 stones.sort((a, b) => a - b);
4 const numberOfStones = stones.length;
5
6 // Initialize minimum moves as number of stones
7 let minimumMoves = numberOfStones;
8
9 // Calculate maximum moves by considering the farthest two stones
10 const maximumMoves = Math.max(
11 stones[numberOfStones - 1] - stones[1],
12 stones[numberOfStones - 2] - stones[0]
13 ) - (numberOfStones - 2);
14
15 // Sliding window to calculate minimum moves needed
16 for (let startIdx = 0, endIdx = 0; endIdx < numberOfStones; ++endIdx) {
17 // While the current window size is greater than slot size, increase the start index
18 while (stones[endIdx] - stones[startIdx] + 1 > numberOfStones) {
19 ++startIdx;
20 }
21
22 // Check for the special case where there's one space for the last stone
23 if (endIdx - startIdx + 1 === numberOfStones - 1 && stones[endIdx] - stones[startIdx] === numberOfStones - 2) {
24 minimumMoves = Math.min(minimumMoves, 2);
25 } else {
26 // Otherwise, calculate the moves needed by the number of stones outside the current window
27 minimumMoves = Math.min(minimumMoves, numberOfStones - (endIdx - startIdx + 1));
28 }
29 }
30
31 // Return an array of minimum and maximum moves
32 return [minimumMoves, maximumMoves];
33}
34
Time and Space Complexity
The time complexity of the given code is O(n log n)
and the space complexity is O(1)
.
Time Complexity
-
stones.sort()
: Sorting the stones requiresO(n log n)
time, wheren
is the number of stones. -
The
for
andwhile
loops: The outerfor
loop runsn
times, and the innerwhile
loop might seem to increase the complexity, but it does not. Thewhile
loop will execute at mostn
times in total across all iterations of thefor
loop because each stone is only moved from the window once. Therefore, the loops combine forO(n)
time.
Adding both complexities, the dominant term is the O(n log n)
for the sort operation, leading to a total time complexity of O(n log n)
.
Space Complexity
- The space complexity remains
O(1)
since no additional space is used that scales with the input size. Only a fixed number of variables are used for tracking indices and the minimum and maximum moves, regardless of the number of stones.
Learn more about how to find time and space complexity quickly using problem constraints.
Which of the following is a min heap?
Recommended Readings
Math for Technical Interviews How much math do I need to know for technical interviews The short answer is about high school level math Computer science is often associated with math and some universities even place their computer science department under the math faculty However the reality is that you
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
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.