[Java][算法 滑动窗口]Day 02---LeetCode 热题 100---08~09

第一题  无重复字符串的最长子串

思路

其实就是在字符串S中 找到没有重复的最长子串的长度  这道题的难点就是在于如何判断最长并且无重复

首先 最长长度  可以使用变量max记录保存   

再者 判断有无重复  最简单的方法就是  暴力遍历法  即对于每次找的子串都再次寻找遍历一次  判断是否已有字符  自然  这种方法判断的话 时间复杂度会不是一般的高

当然 算法优化我们慢慢再讨论  最直接的思路就是如此

解法一:暴力法

我们的暴力当然和上述思路不太一样  我们对于是否重复  可以直接利用Map集合key不能重复的特点  然后使用containKey()方法直接进行判断

class Solution {public static int lengthOfLongestSubstring(String s) {
// 对于S的特殊情况直接返回0if(s==null || s.length()==0) return 0;
//将S转化为char数组[]  遍历该数组 然后进行判断char[] charArray = s.toCharArray();int max=1;Map<Character,Integer> map=new HashMap<>();
//遍历  每次遍历开始就是认定为子串的起始字符for(int i=0;i< charArray.length;i++){
//每次记录完一次后 需要把map清空map.clear();map.put(charArray[i],i); //将第一个字符放入for(int j=i+1;j< charArray.length;j++){if(map.containsKey(charArray[j])){ // 利用containKey方法直接判断if(map.size()>max) max=map.size(); // 判断当前map长度和max进行比较break;}else{map.put(charArray[j],j);}}if(map.size()>max) max=map.size();}return max;}
}

解法二:滑动窗口法

所谓滑动窗口,其实就是在字符串中先选定一段,把这段作为一个可以滑动的窗口,这个窗口类似于队列,每次判断队列中字符是否满足题目条件,不满足即向左滑动(这是整体情况下 一般是向左  自然 也可以只会滑动右边界),滑动窗口的时候,自然左边会少一位,右边会多一位。

class Solution {public int lengthOfLongestSubstring(String s) {if (s.length()==0) return 0;HashMap<Character, Integer> map = new HashMap<Character, Integer>();int max = 0;int left = 0;for(int i = 0; i < s.length(); i ++){if(map.containsKey(s.charAt(i))){ //判断是否存在已有队列中
//有的话 找到map中该重复的元素的索引位置 判断两者索引大小 通过对索引的判断 不断向右滑动变化起点left = Math.max(left,map.get(s.charAt(i)) + 1);  }map.put(s.charAt(i),i);//不存在则添加到map中max = Math.max(max,i-left+1); // 更新最大长度}return max;}
}

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

思路

和第一题类似 也是遍历找子串  只是这个题目判断依据  不是重复 而是找异位词  异位词的定义我们之前也遇到过 简单来说 就是字母组成一样   如果是异位词 那么排序后的字符串两者一定是一样的  所以我们对其的判断方式就是排序后是否一样  那么这道题的解题思路就很明确

需要注意字符串长度  判null等特殊情况

解法一:暴力法

同样的 直接暴力遍历  然后判断是否为异位词就行即可

class Solution {public List<Integer> findAnagrams(String s, String p) {if(s==null||p.length()>s.length()) return new ArrayList<>();char[] pArray = p.toCharArray();Arrays.sort(pArray);ArrayList<Integer> list = new ArrayList<>();int i=0;int l=p.length();//   abvdef   abfor(i=0;i<=s.length()-l;i++){char[] sArray = s.substring(i, i + l).toCharArray();Arrays.sort(sArray);if(String.valueOf(pArray).equals(String.valueOf(sArray))){list.add(i);}}return list;}
}

我们是采用排序的方式进行判断异位词  当然也可以采用哈希表映射 统计字母次数的方式  代码如下

class Solution {public List<Integer> findAnagrams(String s, String p) {int sLen = s.length(), pLen = p.length();if (sLen < pLen) {return new ArrayList<Integer>();}List<Integer> ans = new ArrayList<Integer>();int[] sCount = new int[26];int[] pCount = new int[26];for (int i = 0; i < pLen; ++i) {++sCount[s.charAt(i) - 'a'];++pCount[p.charAt(i) - 'a'];}if (Arrays.equals(sCount, pCount)) {ans.add(0);}for (int i = 0; i < sLen - pLen; ++i) {--sCount[s.charAt(i) - 'a'];++sCount[s.charAt(i + pLen) - 'a'];if (Arrays.equals(sCount, pCount)) {ans.add(i + 1);}}return ans;}
}

解法二:滑动窗口法

我们不再分别统计滑动窗口和字符串 p 中每种字母的数量,而是统计滑动窗口和字符串 p 中每种字母数量的差;并引入变量 differ来记录当前窗口与字符串 p 中数量不同的字母的个数,并在滑动窗口的过程中维护它。

在判断滑动窗口中每种字母的数量与字符串 ppp 中每种字母的数量是否相同时,只需要判断 differ是否为零即可

class Solution {public List<Integer> findAnagrams(String s, String p) {List<Integer> res = new ArrayList<Integer>();int sLen = s.length();int pLen = p.length();if(sLen < pLen) {return res;}//表示字母出现次数差距//count[x] = 0  表示 s与p中字母x出现次数相同 都出现了n次(n>=0)//count[x] = n  表示 在s中字母x出现次数比p多 多出现了n次(n>0)//count[x] = -n 表示 在s中字母x出现次数比p少 少出现了n次(n>0)int[] count = new int[26];for(int i = 0; i < pLen; i++){++count[s.charAt(i)-'a'];--count[p.charAt(i)-'a'];}//表示字母差异个数int differ = 0;for(int j = 0; j < 26; j++){if(count[j]!=0)++differ;}if(differ==0){res.add(0);}//向右滑动for(int i = 0; i < sLen - pLen; i++){//缩减时只考虑count[x]==1与count[x]==0的情况//因为缩减时字母x减少,count[x]会减去1//(1)count[x]==1时(次数差距1次,不相同)  //count[x]==0 -> 次数相同 -> 不相同变相同,字母差异个数减少1 -> differ--//(2)count[x]==0时(次数相同)  //count[x]==-1 -> 次数差距变为1次->相同变不相同 ,字母差异个数增加1 -> differ++//(3)count[x]==-1时(次数不相同) -> count[x]==-2 次数还是不相同-> 字母差异数不变//(4)count[x]==2时(次数不相同)  -> count[x]==1 次数还是不相同-> 字母差异数不变//左缩减一位,iif (count[s.charAt(i) - 'a'] == 1) {  //窗口中s子串左边减少一个s[i]的数量(把原来多出来的1个s[i]去掉,变得相同)//两个字符串字母差距缩小--differ;} else if (count[s.charAt(i) - 'a'] == 0) {  //窗口中s子串左边减少一个s[i]的数量(把原来相同数量的s[i]的减少了1个,数量变得不相同)//两个字符串字母差距增大++differ;}//窗口中s子串左边减少一个字母s[i]--count[s.charAt(i) - 'a'];//添加时只考虑count[x]==-1与count[x]==0的情况,原因分析与缩减时类似//右添加一位,i+pLenif (count[s.charAt(i + pLen) - 'a'] == -1) {  //窗口中s子串右边增加一个s[i+pLen]的数量(把原来缺少的1个s[i]加上,数量变得相同)//两个字符串字母差距缩小--differ;} else if (count[s.charAt(i + pLen) - 'a'] == 0) {  //窗口中s子串右边增加一个s[i+pLen]的数量(把原来相同数量的s[i]多加了1个,变得不相同)//两个字符串字母差距增大++differ;}//窗口中s子串右边增加一个字母s[i+pLen]++count[s.charAt(i + pLen) - 'a'];//两个字符串字母差距为0if (differ == 0) {res.add(i + 1);}}return res;  }
}

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

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

相关文章

操作字符串之子串替换-13-${string//substring/replacement}

1.${string//substring/replacement} 使用$replacement来替换所有匹配的$substring 2.实例 操作字符串样例&#xff1a;stringabc123ABC456xyzabc 字符串操作默认从左边开始进行 命令&#xff1a; echo ${string//abc/ZTJ} [rootkibana ~]# echo ${string//abc/ZTJ} ZTJ…

Validation-参数校验框架

在代码完成后期,为保证数据的正确性和完整性,需要在后端对接口请求的参数进行校验。使用Spring的Validation框架&#xff0c;通过添加注解就可以完成参数校验&#xff0c;不用写if/else来逻辑判断了。 使用该框架&#xff0c;首先在请求参数接收层也就是controller层上方标注V…

【Chrono Engine学习总结】5-sensor-5.1-sensor基础并创建一个lidar

由于Chrono的官方教程在一些细节方面解释的并不清楚&#xff0c;自己做了一些尝试&#xff0c;做学习总结。 1、Sensor模块 Sensor模块是附加模块&#xff0c;需要单独安装。参考&#xff1a;【Chrono Engine学习总结】1-安装配置与程序运行 Sensor Module Tutorial Sensor …

【国产MCU】-CH32V307-基本定时器(BCTM)

基本定时器(BCTM) 文章目录 基本定时器(BCTM)1、基本定时器(BCTM)介绍2、基本定时器驱动API介绍3、基本定时器使用实例CH32V307的基本定时器模块包含一个16 位可自动重装的定时器(TIM6和TIM7),用于计数和在更新新事件产生中断或DMA 请求。 本文将详细介绍如何使用CH32…

苹果Mac键盘如何将 F1 到 F12 取消按Fn

苹果电脑安装了Win10操作系统之后&#xff0c;F1到F12用不了怎么办的解决方法。本文将介绍一些解决方法&#xff0c;帮助您解决无法使用F1到F12功能键的问题。 使用 Mac系统的人都知道&#xff0c;Mac系统默认是没有开启 F1-F12 的使用的&#xff0c;平时我们使用的系统都可以使…

AcWing 802. 区间和 离散化

文章目录 题目链接题目描述解题思路代码实现总结 题目链接 链接: AcWing 802. 区间和 题目描述 解题思路 离散化是一种常用的技巧&#xff0c;它能够将原始的连续数值转换为一组离散的值&#xff0c;从而简化问题的处理。在这段代码中&#xff0c;离散化的过程主要分为三个步…

2024全栈元年-thinkphp-数据操作

thinkphp 数据相关操作 1.单数据查询 1、单数据查询 ,Db::table(‘tp_stu’) 必须加前缀 2、如果只是查询符合条件的使用where find,如果没有符合条件的返回null 3、使用 findOrFail 没有数据会抛出异常 4、使用 findOrEmpty 没有数据会返回【】 5、得到最近一个原生SQL …

C++重新入门-指针篇

C 中的指针是一种非常重要的数据类型&#xff0c;它们存储了内存地址&#xff0c;可以用来直接访问和操作内存中的数据。指针提供了灵活性和直接性&#xff0c;但也需要谨慎使用&#xff0c;因为误用指针可能导致程序崩溃或安全漏洞。以下是对 C 指针的详细解释&#xff1a; 1…

那些也许你不知道的操作符!

前言 操作符有很多种&#xff0c;目前我们已经了解了一部分 例如最简单的、-、*、/、&#xff0c;还有我们学到的&&&#xff0c;||&#xff0c;!等&#xff0c;但是操作符可不是就只有这么些的&#xff0c;让我们一起来看看吧 目录 1. 移位操作符 原码、反码、补码…

当go get获取不到软件包时

当使用go get命令获取软件包时&#xff0c;如果无法成功获取&#xff0c;您可以尝试以下方法来解决问题&#xff1a; 检查网络连接&#xff1a;首先&#xff0c;确保您的计算机能够访问互联网&#xff0c;并且没有任何网络防火墙或代理设置阻止了go get命令的正常运行。 设置代…

Stream Query Denoising for Vectorized HD Map Construction

参考代码&#xff1a;截止2024.02未开源 动机与出发点 这篇文章是在StreamMapNet的基础上做的&#xff0c;为了在局部地图感知任务上提升时序上的感知稳定性&#xff0c;参考DN-DETR中的去噪方案&#xff0c;为局部地图感知提出一种针对局部地图元素的加噪声方案以及去噪逻辑。…

【开源】JAVA+Vue.js实现海南旅游景点推荐系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 用户端2.2 管理员端 三、系统展示四、核心代码4.1 随机景点推荐4.2 景点评价4.3 协同推荐算法4.4 网站登录4.5 查询景点美食 五、免责说明 一、摘要 1.1 项目介绍 基于VueSpringBootMySQL的海南旅游推荐系统&#xff…

深入理解XGBoost:集成学习与堆叠模型

导言 XGBoost是一种强大的集成学习算法&#xff0c;但在解决复杂问题时&#xff0c;单个模型可能无法达到最佳性能。集成学习和堆叠模型是两种有效的方法&#xff0c;可以进一步提高模型的性能。本教程将深入探讨如何在Python中应用集成学习和堆叠模型&#xff0c;使用代码示例…

Linux命令-break命令(结束for,while或until循环。)

说明 结束for&#xff0c;while或until循环&#xff0c;可指定退出几层循环。 语法 break [n]参数 n&#xff08;可选&#xff09;&#xff1a;大于等于1的整数&#xff0c;用于指定退出几层循环。 返回值 返回成功除非n小于1。 示例 # break的可选参数n缺省值为1。 # …

蓝牙BLE学习-安全

1.基本概念 蓝牙标准规定了5种基本的安全服务 身份验证:根据通信设备的蓝牙地址验证其身份。蓝牙不提供本地用户身份验证。保密性:确保只有授权的设备才能访问和查看传输的数据&#xff0c;防止窃听造成的信息泄露。授权(Authorization):在允许设备使用某项服务之前&#xff…

Verilog刷题笔记29

题目&#xff1a; Create a 100-bit binary ripple-carry adder by instantiating 100 full adders. The adder adds two 100-bit numbers and a carry-in to produce a 100-bit sum and carry out. To encourage you to actually instantiate full adders, also output the ca…

C++异常特性以及使用

异常 1.C传统的处理错误方式2.异常概念3.异常使用规则抛出和匹配规则 4.异常的重新抛出4.异常安全5.异常规范6.使用自定义的异常7.C标准异常体系7.异常优缺点 1.C传统的处理错误方式 终止程序&#xff1a;如assert&#xff0c;缺陷&#xff1a;用户难以接受。如发生内存错误&a…

[CUDA 学习笔记] Reduce 算子优化

Reduce 算子优化 注: 本文主要是对文章 【BBuf的CUDA笔记】三&#xff0c;reduce优化入门学习笔记 - 知乎 的学习整理 Reduce 又称之为归约, 即根据数组中的每个元素得到一个输出值, 常见的包括求和(sum)、取最大值(max)、取最小值(min)等. 前言 本文同样按照英伟达官方 PP…

C++ //练习 5.6 改写上一题的程序,使用条件运算符(参见4.7节,第134页)代替if else语句。

C Primer&#xff08;第5版&#xff09; 练习 5.6 练习 5.6 改写上一题的程序&#xff0c;使用条件运算符&#xff08;参见4.7节&#xff0c;第134页&#xff09;代替if else语句。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码…

EasyRecovery软件2024永久绿色免费版下载

EasyRecovery软件在数据恢复领域享有良好的声誉&#xff0c;并且被许多用户认为是一款好用且功能强大的软件。以下是对其功能和用户评价的简要概述&#xff1a; 恢复功能&#xff1a; EasyRecovery提供全面的数据恢复功能&#xff0c;可以恢复因删除、格式化、分区丢失、硬盘故…