从一道Hard学习滑动窗口

Posted 小困

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了从一道Hard学习滑动窗口相关的知识,希望对你有一定的参考价值。

滑动窗口


  滑动窗口(sliding windows algorithm)这种方法,专门用于解决区间解的问题。它在运算的时候,将解集放在窗口中,结束的时候比对是否符合预期。在运算的过程中,会对窗口的左右边缘进行操作(扩大、缩小)。特别针对于线性输入解决,滑动窗口就很形象了。

30. Substring with Concatenation of All Words

  You are given a string, s, and a list of words, words, that are all of the same length. Find all starting indices of substring(s) in s that is a concatenation of each word in words exactly once and without any intervening characters.

  题意是从一串输入字符串s中,找到能全包含words数组的起点(数组元素等长),这个起点可能有多个,而且不需要关心顺序。

Input:
  s = "barfoothefoobarman",
  words = ["foo","bar"]
Output: [0,9]
Explanation: Substrings starting at index 0 and 9 are "barfoo" and "foobar" respectively.
The output order does not matter, returning [9,0] is fine too.

  比如这个例子,他在第0个位置和第9个位置可以包含words数组,也就是0位置和9位置各有一个长度为6的窗口可以囊括words数组。

 

 

Template = "foo","bar"

i=0

[b a r] f o o t h e f o o b a r m a n 

[b a r f o o] t h e f o o b a r m a n 

SAVE

b a r f o o[] t h e f o o b a r m a n 

b a r f o o [t h e] f o o b a r m a n 

b a r f o o t h e [f o o] b a r m a n 

b a r f o o t h e [f o o b a r] m a n 

SAVE

b a r f o o t h e f o o b a r[] m a n 

b a r f o o t h e f o o b a r [m a n] 

i=1

b [a r f] o o t h e f o o b a r m a n 
....

i=2

b a [r f o] o t h e f o o b a r m a n 
....

 

   每次扩展使用单词列表中的词长,如果扩展过程中不符合预期则清除窗口,窗口左端在当前词总数大于词组总数的时候,计算总数是单词的长度(因为s除以单词长度余数是0到单词长度之间,覆盖了余数就能覆盖整个场景)。

public class FindSubstring {
    public void test(){
        // [0,9]
        String s1 = "barfoothefoobarman";
        String[] words1 = {"foo","bar"};
        System.out.println(findSubstring(s1,words1));
        // []
        String s2 = "wordgoodgoodgoodbestword";
        String[] words2  = {"word","good","best","word"};
        System.out.println(findSubstring(s2,words2));
    }
    /**
     *  单词等长
     */
    public List<Integer> findSubstring(String s, String[] words) {
        List<Integer> result = new ArrayList<>();
        if(s == null || words.length == 0){
            return result;
        }
        int step = words[0].length();
        Map<String,Integer> counter = new HashMap<>();
        for(String word :words){
            counter.merge(word, 1, (a, b) -> a + b);
        }

        for(int i=0;i<step;++i){
            Map<String,Integer> window = new HashMap<>();
            int left = i;
            int right = i;
            while (right <= s.length() - step && left <= s.length() - step*words.length){
                String sub = s.substring(right,right + step);
                window.merge(sub,1,(a , b)->a + b);
                if(!counter.containsKey(sub)){
                    window.clear();
                    right += step;
                    left = right;
                    continue;
                }
                while (window.get(sub) > counter.get(sub)){
                    String drop = s.substring(left,left+step);
                    Integer dec = window.get(drop);
                    if(dec != null){
                        if(dec<=1){
                            window.remove(drop);
                        }else {
                            window.put(drop,dec-1);
                        }
                    }
                    left += step;
                }
                right += step;
                if(right - left == step * words.length){
                    result.add(left);
                }
            }
        }
        return result;
    }

}

 

 

 

 76. Minimum Window Substring

Given a string S and a string T, find the minimum window in S which will contain all the characters in T in complexity O(n).

Example:

Input: S = "ADOBECODEBANC", T = "ABC"
Output: "BANC"

题意是将s子串中包含T的所有字母的最短子串找出来,例子中BANC是包含ABC的最短子串。

Template = "ABC"

[]A D O B E C O D E B A N C
[A] D O B E C O D E B A N C
[A D] O B E C O D E B A N C
[A D O] B E C O D E B A N C
[A D O B] E C O D E B A N C
[A D O B E] C O D E B A N C
[A D O B E C] O D E B A N C CONTAINS ABC mark substring A D O B E C ...
[A D O B E C O D E B] A N C B now
2 > 1 so moving left but A is compliance ...
[A D O B E C O D E B A] N C A
2 > 1 moving left
A D O [B E C O D E B A] N C B
2 > 1 moving left
A D O B E [C O D E B A] N C C O D E B A is not shorter than A D O B E C ...
A D O B E [C O D E B A N C] C
2 > 1 moving left
A D O B E C O D E [B A N C] B A N C is shorter than A D O B E C mark substring B A N C

 

import java.util.HashMap;
import java.util.Map;

public class MinimumWindowSubstring {
    public void test(){
        String s1 = "ADOBECODEBANC";
        String t1 = "ABC";
        // BANC
        System.out.println(minWindow(s1,t1));

        String s2 = "a";
        String t2 = "aa";
        // ""
        System.out.println(minWindow(s2,t2));

        String s3 = "a";
        String t3 = "b";
        // ""
        System.out.println(minWindow(s3,t3));

        String s4 = "a";
        String t4 = "a";
        // a
        System.out.println(minWindow(s4,t4));

        String s5 = "ab";
        String t5 = "b";
        // b
        System.out.println(minWindow(s5,t5));

        String s6 = "aa";
        String t6 = "aa";
        // aa
        System.out.println(minWindow(s6,t6));

        String s7 = "acbbaca";
        String t7 = "aba";
        // baca
        System.out.println(minWindow(s7,t7));

        String s8 = "aaaaaaaaaaaabbbbbcdd";
        String t8 = "abcdd";

        System.out.println(minWindow(s8,t8));


    }
    public String minWindow(String s, String t) {
        if(s == null || t == null || s.isEmpty() || t.isEmpty() || s.length() < t.length()){
            return "";
        }
        Map<Character,Integer> counter = new HashMap<>();
        char[] sArray = s.toCharArray();
        for(char c :t.toCharArray()){
            counter.merge(c,1,(a,b)->a+b);
        }
        int left = 0;
        int right ;
        int[] ans = {-1 , 0 , 0};
        Map<Character,Integer> window = new HashMap<>();
        for(int i=0;i<sArray.length;i++){
            char c = sArray[i];
            right = i;
            if(counter.containsKey(c)){
                window.merge(c,1,(a,b)->a+b);
                if(window.keySet().size() == counter.keySet().size()){
                    while (true){
                        Integer dec = window.get(sArray[left]);
                        Integer standard = counter.get(sArray[left]);
                        if(dec != null && standard != null){
                            if(dec <= standard){
                                break;
                            }
                            if(dec > standard){
                                window.merge(sArray[left],-1,(a,b)->a+b);
                            }
                        }
                        left ++;
                    }
                    boolean valid = true;
                    for(char v : counter.keySet()){
                        if(window.get(v) < counter.get(v)){
                            valid = false;
                        }
                    }
                    // mark if initial or shorter sequence
                    if(valid && (ans[0] == -1 || right - left + 1 < ans[0])){
                        ans[0] = right - left + 1;
                        ans[1] = left;
                        ans[2] = right;
                    }
                }
            }
        }
        return ans[0] == -1?"":s.substring(ans[1],ans[2]+1);
    }
}

 239. Sliding Window Maximum

Given an array nums, there is a sliding window of size k which is moving from the very left of the array to the very right. You can only see the k numbers in the window. Each time the sliding window moves right by one position. Return the max sliding window.

 

Input: nums = [1,3,-1,-3,5,3,6,7], and k = 3
Output: [3,3,5,5,6,7] 
Explanation: 

Window position                Max
---------------               -----
[1  3  -1] -3  5  3  6  7       3
 1 [3  -1  -3] 5  3  6  7       3
 1  3 [-1  -3  5] 3  6  7       5
 1  3  -1 [-3  5  3] 6  7       5
 1  3  -1  -3 [5  3  6] 7       6
 1  3  -1  -3  5 [3  6  7]      7

 

class Solution {
    public int[] maxSlidingWindow(int[] nums, int k) {
        List<Integer> result = new ArrayList<>();
        if(nums == null || nums.length == 0){
            return new int[]{};
        }
        int left = 0;
        int windowMax = Integer.MIN_VALUE;
        for(int i=0;i<nums.length;i++){
            if(i - left + 1 > k){
                int rv = nums[i];
                int currentMax = windowMax;
                if(nums[left] == windowMax){
                    currentMax = nums[left+1];
                    for(int t = left+1 ; t <= i ; t ++){
                        currentMax = Math.max(nums[t],currentMax);
                    }
                }
                windowMax = Math.max(currentMax,rv);
                result.add(windowMax);
                left ++ ;
            }  else {
                windowMax = Math.max(nums[i],windowMax);
                if(i - left + 1 == k){
                    result.add(windowMax);
                }
            }
        }
        int[] copyRes = new int[result.size()];
        for(int index = 0 ; index < result.size() ; index ++){
            copyRes[index] = result.get(index);
        }
        return copyRes;
    }
}

 

 





以上是关于从一道Hard学习滑动窗口的主要内容,如果未能解决你的问题,请参考以下文章

滑动窗口/模板单调队列 题解

我的应用程序使用片段,但我需要从左到右从右向左滑动

NC50528 滑动窗口

NC50528 滑动窗口

LeetCode 1838 最高频元素的频数[滑动窗口] HERODING的LeetCode之路

刷题向》一道关于位运算的神题(BZOJ3668)(HARD)