day106(3.7)——leetcode面试经典150

76. 最小覆盖子串

76. 最小覆盖子串

剩下的题都有亿点难度,我还有十道题就写完了,二维动态已经差不多要忘记了,所以我准备先看看思路大概是怎么样的,再写最后一个模块。

今天的这题是之前遗留下来的,我当时觉得很难就没有再写了

出错点:

  1. for(int r=0;r<s.length();r++) {

if(cntTcs\[r]>0 && cntScs\[r]==cntTcs\[r]) {

valid++;

}

今天写了一下大概是用滑动窗口能感觉出来,但是我就是不知道该怎么快速的统计s和t的数,我看了题解,我感觉很新奇,很快速的思路,统计的话,直接用对应的ascll码用数组来统计,快速的判断s的子串是否覆盖t时就:先统计t有多少种字符,然后当每一次右指针滑动的时候,都进行判断,判断此时右指针指向的字符是否属于t需要的一种,并且s子串达标了,s的统计字符+1,

然后就直接可以通过valid和required是否相等,由此可以直接进行缩小范围了。

题目:

题解:

java 复制代码
class Solution {
    public String minWindow(String s, String t) {
         if (s == null || t == null || s.length() < t.length()) {
            return "";
        }
        //由于本题大写字母和小写字母都有,为了方便,代码实现时可以直接创建大小为 128 的数组,保证所有 ASCII 字符都可以统计。这样可以直接统计字符对应的ascll码对应的出现次数
        
        int[] cntS = new int[128];
        int[] cntT = new int[128];
        //先统计字符串t出现次数
        for(char c:t.toCharArray()) {
            cntT[c]++;
        }
        //统计需要多少个requiredValid
        int requiredValid = 0;
        for(int i=0;i<128;i++) {
            if(cntT[i]>0) {
               requiredValid++; 
            }
        }
        //创建记录包含t的最短字符串的左右索引
        int resLeft = -1;   //用来最后判断是否能够得到包含t的字符串
        int resRight = s.length();

        //滑行窗口的左指针,右指针可以直接在遍历s字符串的时候进行遍历
        int l = 0;

        //将s变成char数组,方便统计
        char[] cs = s.toCharArray();
        //创建valid
        int valid = 0;
        //遍历s,不断移动滑动窗口的右边界
        for(int r=0;r<s.length();r++) {
            //右指针次数加1
            cntS[cs[r]]++;
            //看是否符合有效结果,这里必须是==,这样才能使多出来的值不影响达标状态
            if(cntT[cs[r]]>0 && cntS[cs[r]]==cntT[cs[r]]) {
                valid++;
            }

            //说明可以进行缩小范围
            while(valid==requiredValid) {
                if(r-l<resRight-resLeft) {
                    resLeft=l; 
                    //因为substring是左闭右开的begin,end
                    resRight=r+1;
                }
                //将左指针往右移动,缩小范围
                cntS[cs[l]]--;
                //直接将不符合的跳出循环
                if(cntT[cs[l]]>0&&cntS[cs[l]]<cntT[cs[l]]) {
                    valid--;
                }
                l++;
            }
        }
        if(resLeft==-1) {
            return "";
        }
        else {
            return s.substring(resLeft, resRight);
        }
    }
}
相关推荐
南境十里·墨染春水18 分钟前
C++ 工厂模式:从入门到进阶,彻底掌握对象创建的艺术
开发语言·c++·算法
@insist12329 分钟前
系统架构设计师-实时性评价、调度算法与内核架构选型
算法·架构·系统架构·软考·系统架构设计师·软件水平考试
一只齐刘海的猫6 小时前
【Leetcode】找到字符串中所有字母异位词
算法·leetcode·职场和发展
海清河晏1116 小时前
数据结构 | 八大排序
数据结构·算法·排序算法
Frank学习路上7 小时前
【C++】面试:关键字与语法特性
c++·面试
IronMurphy7 小时前
【算法五十七】146. LRU 缓存
算法·缓存
ShineWinsu8 小时前
对于Linux:线程概念与分页存储管理的解析
linux·运维·服务器·面试·线程·进程·虚拟空间地址
文艺倾年8 小时前
【强化学习】强化学习基本概念,20W字总结(一)
人工智能·python·语言模型·自然语言处理·面试·职场和发展·大模型
凌波粒8 小时前
LeetCode--108.将有序数组转换为二叉搜索树(二叉树)
算法·leetcode·职场和发展