含有所有字符的最短字符串之滑动窗口

前言

保持前缀和、滑动窗口对连续子数组和子字符串的敏感性,才能实现问题的转换,以及举一反三的改进。

一、案例

在这里插入图片描述

二、题解

用map记录t每个字符的个数,然后以滑动窗口匹配s,记录的负数表示多出来的,正数表示还缺多少。
每次正数减一,count就加一,加到t的长度就可以比较哈子字符串的大小了;每次begin指针滑动的时候。
map里没有包含begin字符的就不管,否则就看begin字符是处于有无库存的状态,有库存,count不变,无库存,count-1;

package com.xhu.offer.offerII;

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

//含有所有字符的最短字符串
public class MinWindow {
    public String minWindow(String s, String t) {
        //用map记录t每个字符的个数,然后以滑动窗口匹配s,记录的负数表示多出来的,正数表示还缺多少。
        //每次正数减一,count就加一,加到t的长度就可以比较哈子字符串的大小了;每次begin指针滑动的时候。
        // map里没有包含begin字符的就不管,否则就看begin字符是处于有无库存的状态,有库存,count不变,无库存,count-1;
        int len1 = s.length(), len2 = t.length();

        if (len1 < len2) return "";

        int[] res = new int[]{len1 + 1, 0};

        Map<Character, Integer> cache = new HashMap<>();

        for (int i = 0; i < len2; i++) {
            Character ch = t.charAt(i);

            cache.put(ch, cache.getOrDefault(ch, 0) + 1);
        }
        int begin = 0, end = 0, count = 0;

        for (; end < len1; end++) {
            Character ch = s.charAt(end);

            if (!cache.containsKey(ch)) continue;

            int n = cache.get(ch);

            count += n > 0 ? 1 : 0;
            cache.put(ch, n - 1);

            if (count == len2) {

                while (begin <= end) {
                    if (count != len2) break;

                    int len = end - begin + 1;

                    if (len < res[0]) {
                        res[0] = len;
                        res[1] = begin;
                    }

                    Character c = s.charAt(begin);

                    if (cache.containsKey(c)) {
                        int m = cache.get(c);

                        count -= m == 0 ? 1 : 0;

                        cache.put(c, m + 1);
                    }
                    begin++;
                }
            }
        }
        return s.substring(res[1], res[0] == len1 + 1 ? 0 : res[1] + res[0]);
    }
}

总结

1)保持逻辑清晰性,能做来就整理好逻辑,一步一步代码实现,逻辑通不了就别发呆,去看题解。
2)保持前缀和、滑动窗口对连续子数组和子字符串的敏感性,才能实现问题的转换,以及举一反三的改进。

参考文献

[1] LeetCode 原题

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值