算法通关村第十六关—滑动窗口经典问题(白银)

  滑动窗口经典问题

一、最长子串专题

1.1 无重复字符的最长子串

 LeetCode3给定一个字符串s,请你找出其中不含有重复字符的最长子串的长度。例如:

输入:s="abcabcbb"
输出:3
解释:因为无重复字符的最长子串是"abc",所以其长度为3

 要找最长子串,必然要知道无重复字符串的首和尾,然后再从中确定最长的那个,因此至少两个指针才可以,这就想到了滑动窗口思想。
 定义一个K-V形式的map,key表示的是当前正在访问的字符串,value,是其下标索引值。我们每访问一个新元素,都将其下标更新成对应的索引值。具体过程如下图:
image.png
 如果是已经出现过的,例如上述示例中的abcabc,当第二次遇到a时,我们就更新left成为第一个b所在的位置,此时发现left要移动的位置恰好就是map.get(‘a’)+1=1,将’a’用序列来表示,放在一起就是map.get(s.charAt(i)+1。其他情况可以参考图示依次类推。
 有一种特殊情况是需要考虑的,例如abba,我们第二次访问b时, left=map.get('b)+1=2。然后继续访问第二个a,此时left=map.get('a‘)+1=1,也就是left后退了,显然不对。
 我们应该让t在2的基础上继续向前,那该怎么办呢?和原来的对比一下,将最大的加1就可以了,也就是:left=Math.max(left,map.get(s.charAt(i))+1);
完整的代码如下:

class Solution {
    public int lengthOfLongestSubstring(String s) {
        int n = s.length(),max = 0; 
        Map<Character, Integer> map = new HashMap<>(); 
        for(int end = 0, start = 0; end < n; end++){ 
            char element = s.charAt(end); 
            if(map.containsKey(element)){ 
                start = Math.max(map.get(element)+1, start);
            }
            max = Math.max(max, end - start + 1);  
            map.put(element, end);  
        }
        return max;
    }
}

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

 LeetCode159题 给定一个字符串S,找出至多包含两个不同字符的最长子串t,并返回该子串的长度。例如:

输入:"eceba"
输出:3
解释:t是"ece",长度为3

 我们仍然使用Ieft和ight来锁定一个窗口,然后一边向右移动一边分析。我们用一个序列来看一下:aabbcccd。
image.png
 我们接下来需要解决两个问题,一个是怎么判断只有2个元素,另一个是移除的时候怎么知道移除谁,以及移除之后left是什么。
 要判断只有2个元素,还是Hash好用,每一个时刻,这个hashmap包括不超过3个元素。这里还要考虑到要移除谁,所以我们要设计一下Hash的Key-Valuet的含义。我们把字符串里的字符都当做键,在窗口中的最右边的字符位置作为值。此时使用下面的代码就可以确定要删除谁,以及窗口left的新位置:

//伪码,从Hash中找最小值
del_idx = Collections.min(hashmap.values());
left = del_idx + 1;

image.png

public int lengthofLongestSubstringTwoDistinct(String s){
if(s.length() < 3){
    return s.length();
}
int left = 0, right = 0;
HashMap<Character,Integer> hashmap = new HashMap();
int maxLen = 2;
while(right < s.length()){
    if(hashmap.size() < 3)
        hashmap.put(s.charAt(right),right++);
    //如果大小达到了3个
    if(hashmap.size()==3){
        //最左侧要删除的位置
        int del_idx = Collections.min(hashmap.values());
        hashmap.remove(s.charAt(delidx));
        //窗口left的新位置
        left = del_idx + 1;
    }
    maxLen = Math.max(maxLen, right -left);
}
return maxLen;
}

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

 如果再提高一下难度,至多包含K个不同字符的最长子串该怎么办呢?这是LeetCode340题。题目的完整要求是:给定一个字符串S,找出至多包含k个不同字符的最长子串T。示例:

输入:S="eceba",k=2
输出:3
解释:则T"ece",所以长度为3

 本题与上面的题几乎没有区别,只要将判断hash大小为2改成k就可以,超过2就是k+1。十分钟实现:

public int lengthofLongestSubstringTwoDistinct(String s, int k){
if(s.length() < k + 1){
    return s.length();
}
int left = 0, right = 0;
HashMap<Character,Integer> hashmap = new HashMap();
int maxLen = k;
while(right < s.length()){
    if(hashmap.size() < k + 1)
        hashmap.put(s.charAt(right),right++);
    //如果大小达到了k+1个
    if(hashmap.size()== k + 1){
        //最左侧要删除的位置
        int del_idx = Collections.min(hashmap.values());
        hashmap.remove(s.charAt(delidx));
        //窗口left的新位置
        left = del_idx + 1;
    }
    maxLen = Math.max(maxLen, right -left);
}
return maxLen;
}

二、长度最小的子数组

 LeetCode209.长度最小的子数组,给定一个含有n个正整数的数组和一个正整数target。找出该数组中满足其和≥target的长度最小的连续子数组[numsl,numsl±1,numsr-1,numsr],并返回其长度。如果不存在符合条件的子数组,返回0。

输入:target=7,nums=[2,3,1,2,4,3]
输出:2
解释:子数组[4,3]是该条件下的长度最小的子数组。

 本题可以使用双指针来解决,也可以视为队列法,基本思路是先让元素不断入队,当入队元素和等于target时就记录一下此时队列的容量,如果队列元素之和大于target则开始出队,直到小于target则再入队。
 如果出现等于target的情况,则记录一下此时队列的大小,之后继续先入队再出队。每当出现元素之和等于target时我们就保留容量最小的那个。

class Solution {
    public int minSubArrayLen(int target, int[] nums) {
        int left = 0, right = 0;
        int sum = 0;
        int min = nums.length + 1;
        for(right = 0; right < nums.length; right++){
            sum+= nums[right];
            if(sum >= target){
                for(;left < right; left++){
                    if(sum - nums[left]>= target) sum -= nums[left];
                    else break;
                }
                min = Math.min(min, right - left + 1);
            }
        }
        
        return min == nums.length + 1? 0 : min;
    }
}

三.盛水最多的容器

 LeetCode11.给定一个长度为n的整数数组height。有n条垂线,第i条线的两个端点是(i,0)和(i,height[i])。找出其中的两条线,使得它们与×轴共同构成的容器可以容纳最多的水。返回容器可以储存的最大水量。
image.png
平平无奇双指针,下面自己的解法

class Solution {
    public int maxArea(int[] height) {
        int left = 0,right = height.length-1; //创建左右指针,分别指向数组的开端和末端
        int max = 0; //max用于记录最大值
        while(left < right){ //当左右指针还未重合时
            int v = (right - left) * Math.min(height[left] , height[right]); //计算当前的容水量
            max = max > v ? max : v; //利用三目运算符判断最大值是否需要改变
            if(height[left] < height[right]) left++;  //对指向 高度比较低 的指针进行移动
            else right--;
        }
        return max; //返回最大值
    }
}

讲义解法,压缩版

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

四、寻找子串异位词(排列)

4.1 字符串的排列

 LeetCode567.给你两个字符串s1和s2,写一个函数来判断s2是否包含s1的排列。如果是,返回true;否则,返回flse。换句话说,s1的排列之一是s2的子串。其中s1和s2都只包含小写字母。

示例:
输入:s1="ab"s2="eidba00o"
输出:true
解释:s2包含s1的排列之一("ba").
public boolean checkInclusion(String s1, String s2) {
        if(s1.length() > s2.length()) return false;
        int[] arr1 = new int[26];
        int[] arr2 = new int[26];
        for(int i = 0; i < s1.length(); i++){
            arr1[s1.charAt(i) - 'a']++;
            arr2[s2.charAt(i) - 'a']++;
        }
        if(Arrays.equals(arr1,arr2)) return true;
        int j = 0;
        for(int i = s1.length(); i < s2.length(); i++){
             arr2[s2.charAt(i) - 'a']++;
             arr2[s2.charAt(j) - 'a']--;
             j++;
             if(Arrays.equals(arr1,arr2)) return true;
        }
        return false;
    }
}

4.2 找到字符串中所有字母异位

 LeetCode438.找到字符串中所有字母异位词,给定两个字符串s和p,找到s中所有p的异位词的子串,返回这些子串的起始索引。不考虑答案输出的顺序。注意s和ρ仅包含小写字母。异位词指由相同字母重排列形成的字符串(包括相同的字符串)。
例如:

输入:s="cbaebabacd",p="abc"
输出:[0,6]
解释:
起始索引等于0的子串是
"cba",它是"abc"的异位词。
起始索引等于6的子串是"bac",它是"abc"的异位词。

 本题的思路和实现与上面几乎一模一样,唯一不同的是需要用一个List,如果出现异位词,还要记录其开始位置,那直接将其add到List中就可以了。完整代码:

class Solution {
    public List<Integer> findAnagrams(String s, String p) {
        List<Integer> list = new ArrayList();
        if(s.length() < p.length()) return list;
        int[] arr1 = new int[26]; //保存p
        int[] arr2 = new int[26]; //保存s
        for(int i = 0; i < p.length(); i++){
            arr1[p.charAt(i) - 'a']++;
            arr2[s.charAt(i) - 'a']++;
        }
        if(Arrays.equals(arr1,arr2)) list.add(0);
        int left = 0;
        for(int right = p.length(); right < s.length(); right++){
            arr2[s.charAt(right) - 'a']++;
            arr2[s.charAt(left) - 'a']--;
            left++;
            if(Arrays.equals(arr1,arr2)) list.add(left);
        }
        return list;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/324357.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

在Windows中安装MinGW

1、下载 github下载https://github.com/niXman/mingw-builds-binaries/releases 或官网下载https://www.mingw-w64.org/downloads/ 2、选择x86_64-12.1.0-release-posix-seh-rt_v10-rev3 3、解压到当前文件夹 解压之后&#xff0c;可以移动到自己喜欢的文件夹 &#xff0c;复…

1月12日1月15日代码随想录路经总和从中序和后序遍历构造二叉树

112.路经总和 给你二叉树的根节点 root 和一个表示目标和的整数 targetSum 。判断该树中是否存在 根节点到叶子节点 的路径&#xff0c;这条路径上所有节点值相加等于目标和 targetSum 。如果存在&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 叶子节点 …

msvcr120.dll丢失是什么意思,msvcr120.dll丢失怎样修复

msvcr120.dll是一个重要的动态链接库&#xff08;Dynamic Link Library&#xff0c;DLL&#xff09;文件&#xff0c;其中包含了Microsoft Visual C Redistributable for Visual Studio 2013的组件之一。当系统或应用程序提示该DLL文件缺失时&#xff0c;可能会导致应用程序无法…

电商物流查询:未来的发展方向

在电商日益繁荣的时代&#xff0c;物流信息查询不仅关乎消费者体验&#xff0c;更影响着电商运营的效率。快速、准确地追踪物流信息至关重要。本文将简述物流信息快速追踪的价值&#xff0c;并重点介绍固乔快递查询助手这一高效查询工具及其批量查询功能。 一、物流信息快速追踪…

gitLab创建项目无分支,本地新建module提交gitLab教程

第一&#xff1a; gitLab上创建好空项目 第二&#xff1a; 拉取到本地 本地新建module里面的代码拷到文件夹里(把里面的git文件拷到module里面也可以的) 第三&#xff1a; 默认分支为master&#xff0c;本地新建分支release1.0.0 以及 个人的分支feature/1.0.0-*** 新建分支&a…

基于springboot体育场馆运营管理系统源码

基于springboot体育场馆运营管理系统源码330 -- MySQL dump 10.13 Distrib 5.7.31, for Linux (x86_64) -- -- Host: localhost Database: springboot3cprm -- ------------------------------------------------------ -- Server version 5.7.31/*!40101 SET OLD_CHARACT…

Flink 处理函数(1)—— 基本处理函数

在 Flink 的多层 API中&#xff0c;处理函数是最底层的API&#xff0c;是所有转换算子的一个概括性的表达&#xff0c;可以自定义处理逻辑 在处理函数中&#xff0c;我们直面的就是数据流中最基本的元素&#xff1a;数据事件&#xff08;event&#xff09;、状态&#xff08;st…

论文笔记(三十九)Learning Human-to-Robot Handovers from Point Clouds

Learning Human-to-Robot Handovers from Point Clouds 文章概括摘要1. 介绍2. 相关工作3. 背景3.1. 强化学习3.2. 移交模拟基准 4. 方法4.1. Handover Environment4.2. 感知4.3. 基于视觉的控制4.4. 师生两阶段培训 (Two-Stage Teacher-Student Training) 5. 实验5.1. 模拟评估…

智能光栅光片显微成像技术的LabVIEW解决方案

智能光栅光片显微成像技术的LabVIEW解决方案 在生物医学研究中&#xff0c;高效的成像技术对于捕捉细胞内罕见和复杂事件至关重要。智能光栅光片显微技术&#xff08;smartLLSM&#xff09;的出现&#xff0c;代表了LabVIEW软件在高端成像领域的革命性应用&#xff0c;这项技术…

P1563 [NOIP2016 提高组] 玩具谜题————C++

目录 [NOIP2016 提高组] 玩具谜题题目背景题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 样例 #2样例输入 #2样例输出 #2 提示 解题思路Code运行结果 [NOIP2016 提高组] 玩具谜题 题目背景 NOIP2016 提高组 D1T1 题目描述 小南有一套可爱的玩具小人&#xff0c;它…

免费学习鸿蒙(HarmonyOS)开发,一些地址分享

HarmonyOS万物互联&#xff0c;从华为一系列的操作来看已经与iOS、Android形成三足鼎立之势了。 根据《澎湃新闻》的报道&#xff0c;已有23所985高校和46所211高校加入了鸿蒙班的行列&#xff0c;合计达到了69所国内一流高校。通过鸿蒙班的设立&#xff0c;高校可以为学生提供…

4 - JdbcTemplate

spring 框架如何处理对数据库的操作呢? 1. 基本介绍 文档&#xff1a;JdbcTemplate APIs : /spring-framework-5.3.8/docs/javadoc-api/index.html JdbcTemplate 是 Spring 提供的访问数据库的技术。可以将 JDBC 的常用操作封装为模板方法 已经提供了特别多的 API 2. 使用…

【ubuntu】docker中如何ping其他ip或外网

docker中如何ping其他ip或外网 示例图&#xff1a; 运行下面命令&#xff1a; docker run -it --namehei busybox看情况需要加权限 sudo&#xff0c;即&#xff1a; sudo docker run -it --namehei busyboxping 外网 ping -c 4 www.baidu.comping 内网 ping -c 4 192.168.…

steam游戏搬砖项目还能火多久?

最近放假回到老家&#xff0c;见了不少亲戚朋友&#xff0c;大家不约而同都在感叹今年大环境不好&#xff0c;工作不顺&#xff0c;生意效益不好&#xff0c;公司状况不佳&#xff0c;反问我们生意如何&#xff1f;为了让他们心里好受一点&#xff0c;我也假装附和道:也不咋地&…

汽车研发测试大全

车研发中需要做的试验&#xff0c;这些试验都是保证我们的车能安全、稳定、可靠行驶的必要条件。主要包含以下内容&#xff1a; 一、整车试验项目 1.1整车可靠性试验 1.2 NVH试验 1.3 HVAC试验 1.4 EMC试验 1.5 化学分析试验 1.6 整车道路性能试验 二、零部件试验项目 …

node的下载、安装、配置

下载&#xff1a; 官网下载&#xff1a;Node.js 左右两个都可以&#xff1a; 往期版本&#xff1a; ​​​​​​https://registry.npmmirror.com/binary.html?pathnode/v16.18.0/ 其中的16.18.0可以修改成你需要的版本 安装&#xff1a; 打开cmd&#xff1a; 输入以下指令…

精品基于Uniapp+springboot车辆充电桩缴费管理系统管理系统App-地图

《[含文档PPT源码等]精品基于Uniappspringboot充电桩管理系统App》该项目含有源码、文档、PPT、配套开发软件、软件安装教程、项目发布教程、包运行成功&#xff01; 软件开发环境及开发工具&#xff1a; 开发语言&#xff1a;Java 后台框架&#xff1a;springboot、ssm 安…

【STM32单片机】迷宫游戏设计

文章目录 一、主要功能二、软件设计三、实验现象联系作者 一、主要功能 本项目使用STM32F103/F407单片机控制器&#xff0c;TFTLCD触摸屏、按键等。 主要功能&#xff1a; 系统运行后&#xff0c;TFTLCD显示游戏界面&#xff0c;可按下KEY_UP键进入游戏&#xff1b; 系统内置…

持久双向通信网络协议-WebSocket-入门案例实现demo

1 介绍 WebSocket 是基于 TCP 的一种新的网络协议。它实现了浏览器与服务器全双工通信——浏览器和服务器只需要完成一次握手&#xff0c;两者之间就可以创建持久性的连接&#xff0c; 并进行双向数据传输。 HTTP协议和WebSocket协议对比&#xff1a; HTTP是短连接&#xff0…

Javaweb之SpringBootWeb案例员工管理分页查询的详细解析

3. 员工管理 完成了部门管理的功能开发之后&#xff0c;我们进入到下一环节员工管理功能的开发。 基于以上原型&#xff0c;我们可以把员工管理功能分为&#xff1a; 分页查询&#xff08;今天完成&#xff09; 带条件的分页查询&#xff08;今天完成&#xff09; 删除员工&…
最新文章