Sliding Window
Elastic-size Window
The constraint can be expressed as:
\[f(v) \ge 0\]where \(v\) is called contraint variable. Constraint variable is determined by the position and size of the sliding window:
\[v = g(s, m)\]where \(s\) is the start index and \(m\) is the size of the window.
In this type of problems, when \(s\) is fixed and \(m\) increases, \(f(v)\) monotonically increases or decreases. More formally, the following function \(h(m)\) is a monotonic function:
\[f(v) = f(g(m)) = h(m) \ge 0\]Monotonically Decreasing Function
\(h(m)\) is a monotonically decreasing function (MDF). For example, the constraint is “at most k elements”. Denote the number of elements in the sliding window as \(v\), then \(f(v) = k - v \ge 0\). As the window grows, \(v\) tends to increase, so \(f(v)\) decreases.
The common solution: expand the window; whenever the constraint is not satisfied, use a while
loop to shrink the window util the constraint is satisfied again.
Max Length (MDF)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
int longestOnes(vector<int>& nums, int k) {
int i = 0, j = 0, mx = 0;
while (j < nums.size()) {
if (nums[j++] == 0) {
k--;
}
while (k < 0) {
if (nums[i++] == 0) {
k++;
}
}
mx = max(mx, j - i);
}
return mx;
}
An alternative solution is Non-shrinking Window: when the constraint is not satisfied, increment the left pointer i
by one. The sliding window never shrinks.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public int longestOnes(int[] nums, int k) {
int i = 0, j = 0;
// The sliding window never shrinks,
// even if it doesn't meet the requirement at a certain moment
while (j < nums.length) {
if (nums[j++] == 0) {
k--;
}
// If k < 0, both i, j move forward together
// i.e. right shift by one
if (k < 0 && nums[i++] == 0) {
k++;
}
}
// [i, j) is a sliding window.
// Its span memorizes the max range so far
return j - i;
}
The prerequiste of Non-shrinking Window solution is it’s easy to check if the constraint is satisfied or not. Look at the below example. We have to introduce a counter variable to store the number of in-window elements with frequency more than k
.
Length of Longest Subarray With at Most K Frequency
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
int maxSubarrayLength(vector<int>& nums, int k) {
int i = 0, j = 0, len = 0, freqsGtK = 0;
unordered_map<int, int> freqs;
while (j < nums.size()) {
if (freqs[nums[j++]]++ == k) {
freqsGtK++;
}
if (freqsGtK > 0) {
if (--freqs[nums[i++]] == k) {
freqsGtK--;
}
}
}
return j - i;
}
As a comparison, the common solution doesn’t need any extra variables:
1
2
3
4
5
6
7
8
9
10
11
12
int maxSubarrayLength(vector<int>& nums, int k) {
int i = 0, j = 0, len = 0;
unordered_map<int, int> freqs;
while (j < nums.size()) {
freqs[nums[j]]++;
while (freqs[nums[j]] > k) {
freqs[nums[i++]]--;
}
len = max(len, ++j - i);
}
return len;
}
In conclusion, Non-shrinking Window is not always better than the common solution. It depends on the specific problem which one to choose.
Sometimes, we need to convert the problem to the “max consecutive ones” model:
Longest Repeating Character Replacement
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public int characterReplacement(String s, int k) {
int[] freqs = new int[26];
int i = 0, j = 0, maxFreq = 0;
while (j < s.length()) {
// maxFreq: the max occurrences of a char so far; it's non-decreasing.
maxFreq = Math.max(maxFreq, ++freqs[s.charAt(j++) - 'A']);
// Constraint variable: j - i - maxFreq
// After the element s[j - 1] was added to the window in this iteration:
// - If maxFreq remains unchanged. j - i - maxFreq increments by one due to s[j - 1]
// - If maxFreq increases, it must have incremented by one,
// and s[j - 1] must be the new, unique element whose frequency is maxFreq.
// j - i - maxFreqs remains unchanged
//
// Therefore, we can apply the Non-shrinking window technique.
if (j - i - maxFreq > k) {
freqs[s.charAt(i++) - 'A']--;
}
}
return j - i;
}
Frequency of the Most Frequent Element
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public int maxFrequency(int[] nums, int k) {
Arrays.sort(nums);
int i = 0, j = 0;
long availableOps = k;
while (j < nums.length) {
availableOps += nums[j];
// Constraint variable: availableOps - max * length
if (availableOps < (long)nums[j] * (++j - i)) {
availableOps -= nums[i++];
}
}
return j - i;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public int longestNiceSubarray(int[] nums) {
// mask: the ORs of elements in the sliding window
int mask = 0, i = 0, j = 0, max = 0;
while (j < nums.length) {
// Only one element contriutes a set bit in mask
while ((mask & nums[j]) != 0) {
mask ^= nums[i++];
}
mask |= nums[j++];
max = Math.max(max, j - i);
}
return max;
}
Minimum Number of Operations to Make Array Continuous
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
int minOperations(vector<int>& nums) {
ranges::sort(nums);
// De-dupe
int n = nums.size();
nums.erase(unique(nums.begin(), nums.end()), nums.end());
// Finds the longest subarray in which max - min <= n - 1
int i = 0, j = 0;
while (j < nums.size()) {
if (nums[j++] >= nums[i] + n) {
i++;
}
}
// The length of the longest subarray is j - i
// We need to replace n - j - i elements
return n - j + i;
}
Count of Subarrays (MDF)
Count Subarrays With Score Less Than K
1
2
3
4
5
6
7
8
9
10
11
12
public long countSubarrays(int[] nums, long k) {
long sum = 0, count = 0;
int i = 0, j = 0;
while (j < nums.length) {
sum += nums[j++];
while (sum * (j - i) >= k) {
sum -= nums[i++];
}
count += j - i;
}
return count;
}
The following problem applies the At Most K model:
Subarrays with K Different Integers
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
public int subarraysWithKDistinct(int[] nums, int k) {
return atMost(nums, k) - atMost(nums, k - 1);
}
private int atMost(int[] nums, int k) {
int n = nums.length, i = 0, j = 0, result = 0;
int[] freqs = new int[n + 1];
while (j < n) {
if (freqs[nums[j++]]++ == 0) {
k--;
}
while (k < 0) {
if (--freqs[nums[i++]] == 0) {
k++;
}
}
// [i, j) is a sliding window.
// (j - i) represents the count of subarrays that has at most k different integers
// and exclusively ends at index j.
// i.e., subarrays [i, j), [i + 1, j), ..., [j - 1, j)
// Since each loop iteration moves j by one from start to end,
// the final result will include the counts at all positions of j.
result += j - i;
}
return result;
}
An alternative solution is Three pointers, which is a bit more complex.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
int subarraysWithKDistinct(vector<int>& nums, int k) {
// [i1, j) is the longest subarray with k different integers
// [i2, j) is the shortest subarray with k different integers
// Apparently, nums[i2] appears only once in [i2, j)
int n = nums.size(), res = 0, i1 = 0, i2 = 0, j = 0;
vector<int> freqs(n + 1);
while (j < n) {
if (freqs[nums[j++]]++ == 0) {
// In the beginning, the sliding window contains less than k different integers.
// When k == 0, the window contains exactly k different integers.
// After that (k <= 0), the window tends to have more than k different integers with the move of j
// so we need the following block to keep the window having exact k different integers.
if (--k < 0) {
// Since i2 is the only occurrence of nums[i2] until j
// moving i2 right by one will decrement the number of different integers by one
freqs[nums[i2]] = 0;
i1 = ++i2;
}
}
// k never grows.
if (k <= 0) {
// Moves i2 right as far as possible
while (freqs[nums[i2]] > 1) {
freqs[nums[i2++]]--;
}
res += i2 - i1 + 1;
}
}
return res;
}
Count Vowel Substrings of a String
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public int countVowelSubstrings(String word) {
return atMost(word, 5) - atMost(word, 4);
}
private int atMost(String word, int k) {
Map<Character, Integer> freq = new HashMap<>();
int i = 0, j = 0, count = 0;
while (j < word.length()) {
char cj = word.charAt(j++);
// Relocates i if the current char is not vowel
if ("aeiou".indexOf(cj) < 0) {
freq.clear();
i = j;
continue;
}
freq.put(cj, freq.getOrDefault(cj, 0) + 1);
while (freq.size() > k) {
char ci = word.charAt(i++);
freq.put(ci, freq.get(ci) - 1);
freq.remove(ci, 0);
}
count += j - i;
}
return count;
}
1
2
3
4
5
6
7
8
9
10
11
private boolean condition(int[] nums, int upper, int k) {
int i = 0, j = 0, sum = 0, count = 0;
while (j < nums.length) {
sum += nums[j++];
while (sum > upper) {
sum -= nums[i++];
}
count += j - i;
}
return count >= k;
}
Find K-th Smallest Pair Distance
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public int smallestDistancePair(int[] nums, int k) {
Arrays.sort(nums);
int low = 0, high = nums[nums.length - 1] - nums[0];
while (low < high) {
int mid = (low + high) >>> 1;
if (condition(nums, mid, k)) {
high = mid;
} else {
low = mid + 1;
}
}
return low;
}
private boolean condition(int[] nums, int distance, int k) {
int i = 0, j = 0, count = 0;
while (j < nums.length) {
while (nums[j] - nums[i] > distance) {
i++;
}
// It's not `count += j - i` because it's the count of pairs (start != end)
// rather than the count of subarrays (start == end is possible)
count += j++ - i;
}
return count >= k;
}
Delivering Boxes from Storage to Ports
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
int boxDelivering(vector<vector<int>>& boxes, int portsCount, int maxBoxes, int maxWeight) {
int n = boxes.size();
// dp[i]: minimum number of trips to deliver the first i boxes
vector<int> dp(n + 1);
// trips: the number of trips needed to deliver the boxes in the sliding window [i,j)
int i = 0, j = 0, trips = 1, w = 0;
while (j < n) {
// Current port is different from previous port
if (j == 0 || boxes[j][0] != boxes[j - 1][0]) {
trips++;
}
maxWeight -= boxes[j++][1];
// Moves i out of window
// If dp[i] == dp[i + 1], boxes[i] had better be delivered in the previous segment,
// not the current segment due to increasing the total weight
while (j - i > maxBoxes || maxWeight < 0 || (i + 1 < j && dp[i] == dp[i + 1])) {
maxWeight += boxes[i][1];
// Decrements trips if boxes[i] will be delivered to a different port
if (boxes[i][0] != boxes[++i][0]) {
trips--;
}
}
dp[j] = dp[i] + trips;
}
return dp[n];
}
Monotonically Increasing Function
\(h(m)\) is a monotonically increasing function (MIF). For example, the constraint is “sum is greater than or equal to target”. Denote the sum of elements in the sliding window as \(v\), then \(f(v) = v - target \ge 0\). As the window grows, \(v\) tends to increase, so \(f(v)\) increases.
The common solution is opposite compared to that for Monotonically Decreasing Function: expand the window; whenever the constraint is satisfied, use a while
loop to shrink the window util the constraint is not satisfied again.
Min Length (MIF)
1
2
3
4
5
6
7
8
9
10
11
12
13
public int minSubArrayLen(int target, int[] nums) {
int i = 0, j = 0, min = Integer.MAX_VALUE;
while (j < nums.length) {
target -= nums[j++];
while (target <= 0) {
min = Math.min(min, j - i);
target += nums[i++];
}
}
return min == Integer.MAX_VALUE ? 0 : min;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
string minWindow(string s, string t) {
vector<int> freqs(256);
for (char ch : t) {
freqs[ch]++;
}
int i = 0, j = 0, k = t.length(), minLen = numeric_limits<int>::max(), minStart = 0;
while (j < s.length()) {
// Counts of chars in the window:
// * t-chars > 0
// * non-t-chars == 0
if (freqs[s[j++]]-- > 0) {
// Only t-chars will decrement k
k--;
}
while (k == 0) {
if (j - i <= minLen) {
minStart = i;
minLen = j - i;
}
// Counts of chars in the window:
// * t-chars > 0
// * non-t-chars < 0
if (freqs[s[i++]]++ == 0) {
// Only t-chars will increment k
k++;
}
}
}
return minLen == numeric_limits<int>::max() ? "" : s.substr(minStart, minLen);
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public String minWindow(String s1, String s2) {
String window = "";
int i1 = 0, i2 = 0, min = Integer.MAX_VALUE;
while (i1 < s1.length()) {
if (s1.charAt(i1++) == s2.charAt(i2)) {
// pauses when s2 is fully scanned
if (++i2 == s2.length()) {
int j = i1;
// finds the right most i1 in the window that satisfies
// s1.charAt(i1) == s2.charAt(0);
while (--i2 >= 0) {
while (s1.charAt(--i1) != s2.charAt(i2)) {
}
}
i2 = 0;
if (j - i1 < min) {
window = s1.substring(i1, j);
min = j - i1;
}
// moves i1 forward, so we don't get inifite loop
// e.g. "abcde" "bde" (i1 == 1)
i1++;
}
}
}
return window;
}
Count of Subarrays (MIF)
Number of Substrings Containing All Three Characters
This problem is very similar to Minimum Window Substring:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public int numberOfSubstrings(String s) {
int k = 3;
int[] count = new int[k];
int i = 0, j = 0, result = 0;
while (j < s.length()) {
if (count[s.charAt(j++) - 'a']++ == 0) {
k--;
}
while (k == 0) {
if (--count[s.charAt(i++) - 'a'] == 0) {
k++;
}
}
result += i;
}
return result;
}
Count the Number of Good Subarrays
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public long countGood(int[] nums, int k) {
long count = 0;
int i = 0, j = 0;
Map<Integer, Integer> freqs = new HashMap<>();
while (j < nums.length) {
k -= freqs.getOrDefault(nums[j], 0);
freqs.put(nums[j], freqs.getOrDefault(nums[j], 0) + 1);
j++;
while (k <= 0) {
freqs.put(nums[i], freqs.get(nums[i]) - 1);
k += freqs.get(nums[i++]);
}
count += i;
}
return count;
}
Exact-value Constraints
This type of problem can usually be converted to a MDF or MIF problem.
MDF
Minimum Operations to Reduce X to Zero
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public int minOperations(int[] nums, int x) {
int i = 0, j = 0, n = nums.length, sum = Arrays.stream(nums).sum(), min = Integer.MAX_VALUE;
while (j < n) {
// sum(nums[0...i) + nums(j...n - 1]) == x
sum -= nums[j++];
while (sum < x && i < j) {
sum += nums[i++];
}
if (sum == x) {
min = Math.min(min, n - j + i);
}
}
return min == Integer.MAX_VALUE ? -1 : min;
}
Similar to: Maximum Size Subarray Sum Equals k
MIF
Count Complete Subarrays in an Array
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public int countCompleteSubarrays(int[] nums) {
int k = Arrays.stream(nums).boxed().collect(Collectors.toSet()).size(), i = 0, j = 0, result = 0;
Map<Integer, Integer> freqs = new HashMap<>();
while (j < nums.length) {
if (Optional.ofNullable(freqs.put(nums[j], freqs.getOrDefault(nums[j++], 0) + 1)).orElse(0) == 0) {
k--;
}
while (k == 0) {
if (freqs.put(nums[i], freqs.get(nums[i++]) - 1) == 1) {
k++;
}
}
result += i;
}
return result;
}
Non-monotonic Function
\(v = f(s, m)\) is not a monotonic function of \(m\), e.g. “the frequency of each character in the substring is greater than or equal to k”. When you anchor the start index, expanding the window will either make all chars in the window have more than k frequency, or introduce a new char so it doesn’t approach the goal.
Longest Substring with At Least K Repeating Characters
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
public int longestSubstring(String s, int k) {
int max = 0;
// introduces a new constraint `uniqueCharsTarget` so that in sliding window function
// we know when to move i forward (shrink window) without losing possible candidates
for (int uniqueCharsTarget = 1; uniqueCharsTarget <= 26; uniqueCharsTarget++) {
max = Math.max(max, slidingWindow(s, k, uniqueCharsTarget));
}
return max;
}
private int slidingWindow(String s, int k, int uniqueCharsTarget) {
int[] counts = new int[26];
// unique chars in the window
// number of unique chars whose frequency >= k
int uniqueChars = 0, numGeK = 0;
int i = 0, j = 0, max = 0;
while (j < s.length()) {
if (counts[s.charAt(j) - 'a']++ == 0) {
uniqueChars++;
}
if (counts[s.charAt(j++) - 'a'] == k) {
numGeK++;
}
// adjusts the window so that uniqueChars <= uniqueCharsTarget
while (uniqueChars > uniqueCharsTarget) {
if (counts[s.charAt(i) - 'a']-- == k) {
numGeK--;
}
if (counts[s.charAt(i++) - 'a'] == 0) {
uniqueChars--;
}
}
// windows has target number of unique chars and all of them have frequency >= k
if (uniqueChars == uniqueCharsTarget && uniqueChars == numGeK) {
max = Math.max(max, j - i);
}
}
return max;
}
Summary
Monotonically Decreasing Function (MDF) | Monotonically Increasing Function (MIF) | |
---|---|---|
Loop | while (!condition) | while (condition) |
Length | max(j - i) or j - i (Non-shrinking Window) | min(j - i) |
#Subarrays | += j - i | += i |
Fixed-size Window
If the window size is required to be a fixed value k
, then we regard j - i == k
as a constraint, and maintain it in each loop iteration.
Minimum Swaps to Group All 1’s Together
1
2
3
4
5
6
7
8
9
10
11
public int minSwaps(int[] data) {
int i = 0, j = 0, sum = Arrays.stream(data).sum(), count = 0, min = sum;
while (j < data.length) {
count += data[j++];
if (j - i == sum) {
min = Math.min(min, sum - count);
count -= data[i++];
}
}
return min;
}
Maximum Number of Occurrences of a Substring
If a substring occurs n
times, any of its substring occurs at least n
times. So a substring with length minSize
will have the max occurrences.
It’s also common to combine prefix sum to solve this type of problem:
Maximum Points You Can Obtain from Cards
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
vector<int> findAnagrams(string s, string p) {
vector<int> freqs(26);
for (char ch : p) {
freqs[ch - 'a']++;
}
vector<int> v;
int i = 0, j = 0, k = p.length();
while (j < s.length()) {
if (freqs[s[j++] - 'a']-- > 0) {
k--;
}
if (k == 0) {
v.push_back(i);
}
// Count of chars in p won't go below 0
if (j - i == p.length() && freqs[s[i++] - 'a']++ >= 0) {
k++;
}
}
return v;
}
Substring with Concatenation of All Words is similar to this problem, but on word level. Both use the frequency map technique in Minimum Window Substring.
Number of Equal Count Substrings
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public int equalCountSubstrings(String s, int count) {
int result = 0;
int unique = s.chars().mapToObj(i -> (char)i).collect(Collectors.toSet()).size();
for (int k = 1; k <= unique; k++) {
int windowSize = k * count;
int[] c = new int[26];
// count of chars in the window is k
int equalCount = 0;
for (int j = 0; j < s.length(); j++) {
if (++c[s.charAt(j) - 'a'] == count) {
equalCount++;
}
if (j >= windowSize && c[s.charAt(j - windowSize) - 'a']-- == count) {
equalCount--;
}
if (equalCount == k) {
result++;
}
}
}
return result;
}
Minimum Number of Flips to Make the Binary String Alternating
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
int minFlips(string s) {
// Flips needed to maintain "0101..." and "1010..." respectively
vector<int> flips(2);
int n = s.length(), mn = n;
// Sliding window
// Cyclic problem: s += s
for (int i = 0; i < 2 * n; i++) {
// The expected char at i-th index of "0101..."
flips['0' + i % 2 == s[i % n]]++;
// i is the end of the window
if (i >= n) {
// i % n is outside of the window
// Decrements if it was flipped before
flips['0' + (i % n) % 2 == s[i % n]]--;
mn = min(mn, min(flips[0], flips[1]));
}
}
return mn;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public int[] countServers(int n, int[][] logs, int x, int[] queries) {
Arrays.sort(logs, Comparator.comparingInt(l -> l[1]));
int l = logs.length, m = queries.length;
Integer[] indices = new Integer[m];
for (int i = 0; i < m; i++) {
indices[i] = i;
}
Arrays.sort(indices, Comparator.comparingInt(i -> queries[i]));
int[] arr = new int[m];
int i = 0, j = 0, k = 0;
Map<Integer, Integer> freqs = new HashMap<>();
while (k < m) {
while (j < l && logs[j][1] <= queries[indices[k]]) {
freqs.put(logs[j][0], freqs.getOrDefault(logs[j][0], 0) + 1);
j++;
}
while (i < l && logs[i][1] < queries[indices[k]] - x) {
freqs.put(logs[i][0], freqs.get(logs[i][0]) - 1);
freqs.remove(logs[i][0], 0);
i++;
}
arr[indices[k++]] = n - freqs.size();
}
return arr;
}
Minimum Adjacent Swaps for K Consecutive Ones
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
public int minMoves(int[] nums, int k) {
if (k == 1) {
return 0;
}
// Indices of ones
List<Integer> ones = new ArrayList<>();
for (int i = 0; i < nums.length; i++) {
if (nums[i] == 1) {
ones.add(i);
}
}
// Prefix sum
int m = ones.size();
int[] p = new int[m + 1];
for (int i = 0; i < m; i++) {
p[i + 1] = p[i] + ones.get(i);
}
int min = Integer.MAX_VALUE;
// Sliding window ones[i...j] of length k
for (int i = 0, j = k - 1; j < m; i++, j++) {
// Mid point
int mid = (i + j) / 2;
// left: [i, mid - 1], right: [mid + 1, j]
int left = p[mid] - p[i], right = p[j + 1] - p[mid + 1];
// Number of elements on each side
int radius = mid - i;
// For example, ones = [0,1,3,5,7,10,13]
//
// If k is odd, e.g. k = 5, consider the first window [0,1,3,5,7]
// ones[mid] = ones[2] = 3
// left = sum(ones[0,1]) = 1
// right = sum(ones[3,4]) = 12
// radius = 2
//
// Step 1: swaps all ones to the mid
// i.e. [0,1,3,5,7] -> [3,3,3,3,3]
// swaps1 = (3 - 0) + (3 - 1) + (5 - 3) + (7 - 3)
// = right - left
// Step 2: distributes the ones at the mid uniformly in the window
// i.e. [3,3,3,3,3] -> [1,2,3,4,5]
// swaps2 = (3 - 1) + (2 - 1) + (4 - 3) + (5 - 3)
// = 2 * (1 + 2) = (1 + radius) * radius
// = subtrahend
// swap = swaps1 - swaps2 = left + right - subtrahend
//
// Similarly, if k is even, e.g. k = 6, consider the first window [0,1,3,5,7,10]
// ones[mid] = ones[2] = 3
// left = sum(ones[0,1]) = 1
// right = sum(ones[3,5]) = 22
// radius = 2
//
// Step 1: [0,1,3,5,7,10] -> [3,3,3,3,3,3]
// swaps1 = right - left - 3 = right - left - ones[mid]
// Step 2: [3,3,3,3,3,3] -> [1,2,3,4,5,6]
// swaps2 = 2 + 1 + 0 + 1 + 2 + 3
// = (1 + radius) * radius + (radius + 1)
int subtrahend = radius * (radius + 1) + (k % 2 == 0 ? radius + 1 + ones.get(mid) : 0);
min = Math.min(min, right - left - subtrahend);
}
return min;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public boolean canReach(String s, int minJump, int maxJump) {
// prev: the number of previous positions that we can jump from
int n = s.length(), prev = 0;
boolean[] dp = new boolean[n];
dp[0] = true;
for (int i = 1; i < n; i++) {
// Sliding window: [i - maxJump, i - minJump]
if (i >= minJump && dp[i - minJump]) {
prev++;
}
if (i > maxJump && dp[i - maxJump - 1]) {
prev--;
}
// Checks if there's a true in the window
dp[i] = prev > 0 && s.charAt(i) == '0';
}
return dp[n - 1];
}
Minimum Number of K Consecutive Bit Flips
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
int minKBitFlips(vector<int>& nums, int k) {
// accumulated: parity of the number of flips at nums[i] caused by the sliding window
// nums[i - k + 1, ..., i - 1]
int accumulated = 0, flips = 0, n = nums.size();
// Checks if the k-sized window starting from i needs flipping
for (int i = 0; i < n; i++) {
if (i >= k && nums[i - k] > 1) {
accumulated ^= 1;
nums[i - k] -= 2;
}
// Needs flipping
if (accumulated == nums[i]) {
// The max window starting from i is less than k, so the flipping cannot be done
if (i + k > n) {
return -1;
}
// A lazy way to mark nums[i] is flipped:
// if nums[i] is not binary, it's flipped; otherwise not flipped
nums[i] += 2;
accumulated ^= 1;
flips++;
}
}
return flips;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
public int kEmptySlots(int[] bulbs, int k) {
int n = bulbs.length;
// days[i]: the day when bulbs[i] is turned on (1-indexed)
int[] days = new int[n];
for (int i = 0; i < n; i++) {
days[bulbs[i] - 1] = i + 1;
}
// sliding window
// the goal is find a fixed window in days, whose length is (k + 2)
// and for all indexes in between (exclusively), days[index] > endpoints
int min = Integer.MAX_VALUE;
int left = 0, right = k + 1;
for (int i = 0; right < n; i++) {
// current days[i] is valid, continue
if (days[i] > days[left] && days[i] > days[right]) {
continue;
}
// reaches the right endpoint
// since all previous number are valid, this is a candidate minimum
if (i == right) {
min = Math.min(min, Math.max(days[left], days[right]));
}
// not valid, slides the window
left = i;
right = k + 1 + i;
}
return min == Integer.MAX_VALUE ? -1 : min;
}
Quasi-fixed Window
The sliding window is close to fixed, but not exactly.
Moving Stones Until Consecutive II
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
vector<int> numMovesStonesII(vector<int>& stones) {
ranges::sort(stones);
int n = stones.size();
int i = 0, j = 0, mn = n;
while (j < n) {
// The sliding window is tight, i.e. the ends of the window are always on stone positions.
// Window size: stones[j] - stones[i] + 1
// Number of stones in the window: j - i + 1
// Moves i so that the window size <= n and is closest to n
while (stones[j] - stones[i] >= n) {
i++;
}
// Corner case:
// * number of stones in the window is (n - 1)
// * window size is (n - 1)
// e.g. [1,2,3,4,10] -> [2,3,4,6,10] -> [2,3,4,5,6]
//
// The logic is valid even for the the following example:
// e.g. [1,2,3,4,6] -> [2,3,4,5,6]
// * First window (i = 0, j = 3) matches the corner case, mn = 2;
// * Second window (i = 1, j = 4) falls into the else block, mn = 1
if (j - i == n - 2 && stones[j] - stones[i] == n - 2) {
mn = min(mn, 2);
} else {
// e.g. [1,2,4,5,10] -> [1,2,3,4,5]
mn = min(mn, n - (j - i + 1));
}
j++;
}
// Moves leftmost or rightmost stone
// e.g. moves leftmost to the next available slot
// [1,3,5,10] -> [3,4,5,10] -> [4,5,6,10]
//
// Max of avaible slots:
// - left -> right: (stones[n - 1] - stones[1] + 1) - (n - 1)
// - right -> left: (stones[n - 2] - stones[0] + 1) - (n - 1)
int mx = max(stones[n - 1] - stones[1], stones[n - 2] - stones[0]) - n + 2;
return {mn, mx};
}
Special Cases
Array Value as Window Boundaries
In these problems, array values are used to udpate window boundaries, and distances between array indices are used to update quantities in window.
Maximize Win From Two Segments
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public int maximizeWin(int[] prizePositions, int k) {
int n = prizePositions.length;
// dp[i]: In the first i positions, the maximum number of prizes we can get from one segment
int[] dp = new int[n + 1];
int i = 0, j = 0, max = 0;
while (j < n) {
while (prizePositions[i] + k < prizePositions[j]) {
i++;
}
j++;
// The prize at index j is either selected or not
dp[j] = Math.max(dp[j - 1], j - i);
max = Math.max(max, j - i + dp[i]);
}
return max;
}
Maximum White Tiles Covered by a Carpet
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public int maximumWhiteTiles(int[][] tiles, int carpetLen) {
Arrays.sort(tiles, Comparator.comparingInt(t -> t[0]));
// It's always optimal to align the carpet with the left of a tile range.
// i, j are indices of tiles array, not the actual tile position.
int i = 0, j = 0, max = 0, cover = 0;
while (max < carpetLen && j < tiles.length) {
if (i == j || tiles[i][0] + carpetLen > tiles[j][1]) {
// case 1: tiles[j] is the first tile (i == j)
// carpet may be longer or shorter than this tile
// so picks the min of the two as the covered length
// case 2: carpet fully covers tiles[j]
//
// in either case, moves tiles[j] into the window
cover += Math.min(carpetLen, tiles[j][1] - tiles[j][0] + 1);
max = Math.max(max, cover);
j++;
} else {
// Partial of tiles[j] is covered by the carpet
int partial = Math.max(0, tiles[i][0] + carpetLen - tiles[j][0]);
max = Math.max(max, cover + partial);
// Moves tile[i] out of the window.
// tiles[i] always aligns with the carpet start,
// so it's always fully covered by the carpet in this else-branch.
cover -= tiles[i][1] - tiles[i][0] + 1;
i++;
}
}
return max;
}