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

最长字串专题

无重复字符的最长子串

3. 无重复字符的最长子串 - 力扣(LeetCode)

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

示例 1:

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

示例 2:

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

示例 3:

输入: s = "pwwkew"
输出: 3
解释: 因为无重复字符的最长子串是 "wke",所以其长度为 3。请注意,你的答案必须是 子串 的长度,"pwke" 是一个子序列,不是子串。

提示:

  • 0 <= s.length <= 5 * 104
  • s 由英文字母、数字、符号和空格组成
滑动窗口+HashMap
  1. start不动,end向后移动
  2. end遇到重复字符时,若重复字符的上一个位置比start靠右,则start更新为重复字符的上一个位置 + 1(即子串从不重复的位置重新开始统计)。同时记录最长字串的长度
  3. hashMap的key记录字符,value记录重复字符的最新下标。
public int lengthOfLongestSubstring(String s)
{int start = 0;int ans = 0;HashMap<Character, Integer> hashMap = new HashMap<Character, Integer>();for (int end= 0; end < s.length(); end++){char c =  s.charAt(end);if ((hashMap.containsKey(c)))start = Math.max(hashMap.get(c) + 1, start);hashMap.put(c, end);ans = Math.max(ans, end - start + 1);}return ans;
}

参考题解:3. 无重复字符的最长子串 - 力扣(LeetCode)

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

159. 至多包含两个不同字符的最长子串 - 力扣(LeetCode)

给你一个字符串 s ,请你找出 至多 包含 两个不同字符 的最长子串,并返回该子串的长度。

示例 1:

输入:s = "eceba"
输出:3
解释:满足题目要求的子串是 "ece" ,长度为 3

示例 2:

输入:s = "ccaabbb"
输出:5
解释:满足题目要求的子串是 "aabbb" ,长度为 5
基于HashMap的滑动窗口
public int lengthOfLongestSubstringTwoDistinct(String s)
{if (s.length() < 3)return s.length();// hashMap 中的字符 -> 字符在滑动窗口中最靠右的位置HashMap<Character, Integer> hashMap = new HashMap<Character, Integer>();int ans = 2;for (int left = 0, right = 0; right < s.length(); right++){hashMap.put(s.charAt(right), right);//滑动窗口中不同字符超过2个(应维持在2个及以内)if(hashMap.size() > 2){//提取滑动窗口中最靠左的字符的位置int t = Collections.min(hashMap.values());//删除在滑动窗口中删除该字符hashMap.remove(s.charAt(t));//更新滑动窗口中左侧指针left = t + 1;}ans = Math.max(ans, right - left + 1);}return ans;
}

参考题解:159. 至多包含两个不同字符的最长子串 - 力扣(LeetCode)

拓展题

340. 至多包含 K 个不同字符的最长子串 - 力扣(LeetCode)

给你一个字符串 s 和一个整数 k ,请你找出 至多 包含 k不同 字符的最长子串,并返回该子串的长度。

public int lengthOfLongestSubstringKDistinct(String s, int k)
{if (s.length() < k)return s.length();// hashMap 中的字符 -> 字符在滑动窗口中最靠右的位置HashMap<Character, Integer> hashMap = new HashMap<Character, Integer>();int ans = k;for (int left = 0, right = 0; right < s.length(); right++){hashMap.put(s.charAt(right), right);//滑动窗口中不同字符超过k个(应维持在k个及以内)if(hashMap.size() > k){//提取滑动窗口中最靠左的字符的位置int t = Collections.min(hashMap.values());//删除在滑动窗口中删除该字符hashMap.remove(s.charAt(t));//更新滑动窗口中左侧指针left = t + 1;}ans = Math.max(ans, right - left + 1);}return ans;
}

长度最小的子数组

209. 长度最小的子数组 - 力扣(LeetCode)

给定一个含有 n 个正整数的数组和一个正整数 target

找出该数组中满足其总和大于等于 target 的长度最小的 连续子数组 [numsl, numsl+1, ..., numsr-1, numsr] ,并返回其长度**。**如果不存在符合条件的子数组,返回 0

示例 1:

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

示例 2:

输入:target = 4, nums = [1,4,4]
输出:1

示例 3:

输入:target = 11, nums = [1,1,1,1,1,1,1,1]
输出:0

提示:

  • 1 <= target <= 109
  • 1 <= nums.length <= 105
  • 1 <= nums[i] <= 105

大小可变的滑动窗口

这里用队列的结构来演示
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

public int minSubArrayLen(int target, int[] nums)
{int ans = Integer.MAX_VALUE;int sum = 0;for (int left = 0, right = 0; right < nums.length; right++){//滑动窗口向右延伸sum += nums[right];//窗口中所有元素之和>=target时,记录长度,并收缩窗口左侧while (sum >= target){ans = Math.min(ans, right - left + 1);//收缩sum -= nums[left];left++;}}//存在[nums的所有元素之和 < target]的情况,这种情况下ans无法更新return ans == Integer.MAX_VALUE ? 0 : ans;
}

盛最多水的容器

11. 盛最多水的容器 - 力扣(LeetCode)

给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0)(i, height[i])

找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。

返回容器可以储存的最大水量。

**说明:**你不能倾斜容器。

示例 1:

img

输入:[1,8,6,2,5,4,8,3,7]
输出:49 
解释:图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49

示例 2:

输入:height = [1,1]
输出:1

提示:

  • n == height.length
  • 2 <= n <= 105
  • 0 <= height[i] <= 104

向内收缩

盛水容量(面积) = 短板高度 * 两板距离。初始化一个板在最左,一个板在最右。向内收缩的过程中,两板距离一定减小,收缩短板一侧可能使短板变长板,原来的长板成为新的短板,从而使面积增大;但收缩长板一侧并不能改变原来的短板,甚至可能会使长板变短板,导致面积进一步减小。

所以我们只有在遍历的过程中,不断收缩短板一侧即可,同时记录最大盛水容量(面积)

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

寻找字串异位词

字符串的排列

567. 字符串的排列 - 力扣(LeetCode)

给你两个字符串 s1s2 ,写一个函数来判断 s2 是否包含 s1 的排列。如果是,返回 true ;否则,返回 false

换句话说,s1 的排列之一是 s2子串

示例 1:

输入:s1 = "ab" s2 = "eidbaooo"
输出:true
解释:s2 包含 s1 的排列之一 ("ba").

示例 2:

输入:s1= "ab" s2 = "eidboaoo"
输出:false

提示:

  • 1 <= s1.length, s2.length <= 104
  • s1s2 仅包含小写字母
大小固定的滑动窗口

参考题解:567. 字符串的排列 - 力扣(LeetCode)

public boolean checkInclusion(String s1, String s2)
{//countS1用于统计s1中各字母的出现次数int[] countS1 = new int[26];for (int i = 0; i < s1.length(); i++)countS1[s1.charAt(i) - 'a']++;//countS2用于统计滑动窗口(大小固定)中各字母出现次数int[] countS2 = new int[26];int left = 0, right = 0;while (right < s2.length()){//窗口移动countS2[s2.charAt(right) - 'a']++;right++;//判断依据:窗口内字母的出现次数等于s1中字母的出现次数if (Arrays.equals(countS1, countS2))return true;//滑动窗口中的元素数量超过限定值的一个//删去最左侧的元素if (right - left == s1.length()){countS2[s2.charAt(left) - 'a']--;left++;}}return false;
}

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

438. 找到字符串中所有字母异位词 - 力扣(LeetCode)

给定两个字符串 sp,找到 s 中所有 p异位词 的子串,返回这些子串的起始索引。不考虑答案输出的顺序。

异位词 指由相同字母重排列形成的字符串(包括相同的字符串)。

示例 1:

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

示例 2:

输入: s = "abab", p = "ab"
输出: [0,1,2]
解释:
起始索引等于 0 的子串是 "ab", 它是 "ab" 的异位词。
起始索引等于 1 的子串是 "ba", 它是 "ab" 的异位词。
起始索引等于 2 的子串是 "ab", 它是 "ab" 的异位词。

提示:

  • 1 <= s.length, p.length <= 3 * 104
  • sp 仅包含小写字母
用一个list记录索引即可
public List<Integer> findAnagrams(String s, String p)
{//countS1用于统计p中各字母的出现次数int[] countS1 = new int[26];for (int i = 0; i < p.length(); i++)countS1[p.charAt(i) - 'a']++;//countS2用于统计滑动窗口(大小固定)中各字母出现次数int[] countS2 = new int[26];int left = 0, right = 0;ArrayList<Integer> ans = new ArrayList<>();while (right < s.length()){//窗口移动countS2[s.charAt(right) - 'a']++;right++;if (Arrays.equals(countS1, countS2))ans.add(left);//滑动窗口中的元素数量超过限定值的一个//删去最左侧的元素if (right - left == p.length()){countS2[s.charAt(left) - 'a']--;left++;}}return ans;
}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ChatGPT高效提问—prompt基础

ChatGPT高效提问—prompt基础 ​ 设计一个好的prompt对于获取理想的生成结果至关重要。通过选择合适的关键词、提供明确的上下文、设置特定的约束条件&#xff0c;可以引导模型生成符合预期的回复。例如&#xff0c;在对话中&#xff0c;可以使用明确的问题或陈述引导模型生成…

1.0 Zookeeper 分布式配置服务教程

ZooKeeper 是 Apache 软件基金会的一个软件项目&#xff0c;它为大型分布式计算提供开源的分布式配置服务、同步服务和命名注册。 ZooKeeper 的架构通过冗余服务实现高可用性。 Zookeeper 的设计目标是将那些复杂且容易出错的分布式一致性服务封装起来&#xff0c;构成一个高…

远程控制APP,你的远程控制专家

在这个科技日新月异的时代&#xff0c;我们的生活被各种手机软件所包围。几乎每个人都有一个甚至多个手机&#xff0c;你是否也有遇到过需要远程操作自己某一台手机的场景呢&#xff1f;今天&#xff0c;我要向大家推荐一款神奇的手机远程操作神器&#xff0c;让你可以随时随地…

windows集成的hyper-v三种网络形式

windows集成了hyper-v,有人反应不习惯使用hyper-v,hyper-v的一些使用习惯确实异于vmware的workstation,比如说hyper-v的三种虚拟交换机&#xff0c;和VMware worksation网络模式叫法有很大差异&#xff0c;这篇文章主要想谈谈hyper-v的三种虚拟交换机以及适应场景。 如图所示&…

Python中的for循环用法详解,一文搞定它

文章目录 for循环1.for循环的基本语法&#xff08;1&#xff09;遍历不等长多级容器&#xff08;2&#xff09;遍历不等长多级容器&#xff08;3&#xff09;遍历等长的容器 2.变量的解包3.for...else【详细讲解】4.range对象5.总结6.打印 1 ~ 10 跳过57.打印菱形小星星 for循环…

大众日报《大众日报》投稿要求//投稿邮箱

大众日报《大众日报》投稿要求 《大众日报》2000字符&#xff0c;1个月内&#xff0c;可加急。 《大众日报》教育纯新闻稿&#xff0c;500字左右 《大众日报》理论版&#xff0c;主要收思政稿&#xff0c;先看稿子。2000字符&#xff1b;有稿最快一周 山东党机关报纸《大众日…

感悟笔记——2024年2月5日

今日阅读了一篇挺有深度的文章&#xff0c;主要阐述进入职场后的大部分人&#xff0c;是怎么逐渐沦为螺丝钉的?即使起点巨高的优等生&#xff0c;也不可避免。文章指路&#xff1a; 「优等生思维」正在将你变成「螺丝钉」和「老黄牛」从小到大&#xff0c;我一直都是那个「别…

【Java 数据结构】泛型进阶

泛型 1 什么是泛型2 引出泛型2.1 语法 3 泛型类的使用3.1 语法3.2 示例3.3 类型推导(Type Inference) 泛型是如何编译的擦除机制裸类型4 泛型的上界4.1 语法4.2 示例4.3 复杂示例 5 泛型方法5.1 定义语法5.2 示例5.3 使用示例-可以类型推导5.4 使用示例-不使用类型推导 6 通配符…

AI嵌入式K210项目(26)-二维码识别

文章目录 前言一、什么是二维码&#xff1f;二、实验准备三、实验过程四、API接口总结 前言 本章介绍基于机器视觉实现二维码识别&#xff0c;主要包含两个过程&#xff0c;首先检测图像中是否有二维码&#xff0c;如果有则框出并打印二维码信息&#xff1b; 一、什么是二维码…

分享65个节日PPT,总有一款适合您

分享65个节日PPT&#xff0c;总有一款适合您 65个节日PPT下载链接&#xff1a;https://pan.baidu.com/s/1hc1M5gfYK8eDxQVsK8O9xQ?pwd8888 提取码&#xff1a;8888 Python采集代码下载链接&#xff1a;采集代码.zip - 蓝奏云 学习知识费力气&#xff0c;收集整理更不易。知…

AE2023 After Effects 2023

After Effects 2023是一款非常强大的视频编辑软件&#xff0c;提供了许多新功能和改进&#xff0c;使得视频编辑和合成更加高效和灵活。以下是一些After Effects 2023的特色功能&#xff1a; 新合成预设列表&#xff1a;After Effects 2023彻底修改了预设列表&#xff0c;使其…

VSTO打包Word插件WPS也支持

启动AdvancedInstallerPortable.exe打包软件 选择“加载项” 选择“office加载项”之后点“创建项目” 四、输入自已的插件名和公司名 任选一种包类型 五、选择包的保存位置 勾选“vsto office加载项” 六、选择要打包的项目debug文件夹 选择相应版本 配置相应环境 选择语言 添…