Post

Sliding Window

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)

Max Consecutive Ones III

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;
}

Longest Nice Subarray

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;
}

Kth Smallest Subarray Sum

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)

Minimum Size Subarray Sum

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;
}

Minimum Window Substring

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);
}

Minimum Window Subsequence

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)
Loopwhile (!condition)while (condition)
Lengthmax(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

Find All Anagrams in 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
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;
}

Count Zero Request Servers

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;
}

Jump Game VII

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;
}

K Empty Slots

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;
}
This post is licensed under CC BY 4.0 by the author.