算法通关村——滑动窗口高频问题

1. 无重复字符的最长子串

给定一个字符串 s ,请你找出其中不含有重复字符的 最长子串 的长度。
示例 1:
输入: s = “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。

1.1 滑动窗口

找到最长字串需要找到字串的首尾位置,而这个首尾位置就是滑动窗口的大小。题目中出现了重复,一般是想到使用hashmap作为存储元素,而这个里面可以将字符串的相应的字符作为k,然后它的下标作为value,一旦遇到有重复字符的,就将左窗口移到当前的位置的右边一位,作为新的字符串的开始位置。

算法通关村——滑动窗口高频问题_第1张图片

    public int lengthOfLongestSubstring(String s) {
        if(s.length() == 0) return 0;
        // 左窗口开始
        int left = 0;
        int max = 0;
        // 存放k:字符,v:字符下标
        HashMap<Character,Integer> hashMap = new HashMap<>();
        for(int right = 0;right<s.length();right++){
            // 里面包含当前字符,需要移动左窗口的位置
            if(hashMap.containsKey(s.charAt(right))){
                left = Math.max(left,hashMap.get(s.charAt(right))+1);
            }
            // 不包含,就将元素和它的下标添加
            hashMap.put(s.charAt(right),right);
            // 比较大小
            max = Math.max(max,right-left+1);
        }
        return max;
    }

2. 至多包含两个不同字符的最长子串

给定一个字符串 s ,找出 至多 包含两个不同字符的最长子串 t ,并返回该子串的长度,这就是LeetCode159题。例如:
输入: “eceba”
输出: 3
解释: t 是 “ece”,长度为3。

2.1 滑动窗口

字符串依然是需要首尾位置,采用滑动窗口,和第一题类似,不过这一题的关键在于字符串是两个不同的字符,判断字符是否重复可以使用hashmap,而问题在于如何判断只有两个字符,出现第三个字符应该如何去除之前的字符。
可以使用 int del_idx = Collections.min(hashmap.values());
hashmap.remove(s.charAt(del_idx));
算法通关村——滑动窗口高频问题_第2张图片

public int lengthOfLongestSubstringTwoDistinct(String s) {
        if (s.length() <= 2) {
            return s.length();
        }
        // 最大长度
        int maxLength = 2;
        int left = 0;
        int right = 0;

        HashMap<Character, Integer> map = new HashMap<>();
        while (right < s.length()) {
            // map长度小于3,意味着还没出现第三个不一样的字符,此时可以继续添加,更改对于元素下标
            if(map.size() < 3) {
             map.put(s.charAt(right), right++);
            }
            // map长度等于3,此时出现了第三个不一样的字符,需要删除一个最小位置下标的字符,然后移动左指针到当前删除下标的下一个位置
            if(map.size() == 3){
                int del_idx = Collections.min(map.values());
                map.remove(s.charAt(del_idx));
                left = del_idx + 1;
            }
            maxLength = Math.max(maxLength, right - left);
        }
        return maxLength;
    }

3. 至多包含 K 个不同字符的最长子串

LeetCode340题。
题目的完整要求是:给定一个字符串 s,找出 至多 包含 k 个不同字符的最长子串T。示例:
输入: s = “eceba”, k = 2
输出: 3
解释: 则 T 为 “ece”,所以长度为 3。

3.1 滑动窗口

这一题和上面一样,只不过是将2替换成了k,判断的是否改成k就可以。

    public int lengthOfLongestSubstringKDistinct(String s, int k) {
        if (s.length() < k) return 0;
        int left = 0;
        int right = 0;
        HashMap<Character, Integer> map = new HashMap<>();
        int maxLength = k;
        while (right < s.length()) {
            if (map.size() < k+1) {
                map.put(s.charAt(right), right++);
            }
            if(map.size() == k+1){
                int del_idx = Collections.min(map.values());
                map.remove(s.charAt(del_idx));
                left = del_idx + 1;
            }
            maxLength = Math.max(maxLength, right - left);
        }
        return maxLength;
    }

4. 长度最小的子数组

长度最小的子数组
给定一个含有 n 个正整数的数组和一个正整数 target 。

找出该数组中满足其和 ≥ target 的长度最小的 连续子数组 [numsl, numsl+1, …, numsr-1, numsr] ,并返回其长度。如果不存在符合条件的子数组,返回 0 。
示例 1:
输入:target = 7, nums = [2,3,1,2,4,3]
输出:2
解释:子数组 [4,3] 是该条件下的长度最小的子数组。

4.1 滑动窗口

这一题主要思路就是计算滑动窗口区间内的元素和,如果元素和小于目标值,可以继续添加,如果元素和大于目标值,那么就需要将左窗口位置的元素从元素和里面去除,然后移动指针,直到元素和小于目标值。

  public int minSubArrayLen(int target, int[] nums) {
        int left =0;
        int right =0;
        int minLength = Integer.MAX_VALUE;
        int sum = 0;
        while(right<nums.length){
            sum += nums[right++];
            while(sum >= target){
                minLength = Math.min(minLength,right-left);
                sum = sum -  nums[left++];
            }
        }

        return minLength == Integer.MAX_VALUE ? 0 : minLength;
    }

5. 盛最多水的容器

盛最多水的容器
给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。
找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。
返回容器可以储存的最大水量。
说明:你不能倾斜容器。

输入:[1,8,6,2,5,4,8,3,7]
输出:49

5.1 滑动窗口

这一题思路还是很清晰的,首先定义两个指针分别指向首尾,然后向中间移动,移动过程中需要判断左右两个高度,选出最短的,然后计算面积。

 public int maxArea(int[] height) {
        int left = 0;
        int right = height.length - 1;
        int maxAre = 0;
        while(left < right){
            maxAre = 
            height[left] < height[right] ? 
            Math.max(maxAre,(right-left) * height[left++] ):
            Math.max(maxAre,(right-left) * height[right--]);
        }
        return maxAre;
    }

你可能感兴趣的:(算法,算法)