【力扣高频题】003.无重复字符的最长子串

2024-06-10 15:20

本文主要是介绍【力扣高频题】003.无重复字符的最长子串,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前段时间和小米的某面试官聊天。因为我一直在做 算法文章 的更新,就多聊了几句算法方面的知识。

并且在聊天过程中获得了一个“重要情报”:只要他来面试,基本上每次的算法题,都会去考察关于 子串和子序列 的问题。

的确,如果说哪种算法更容易在面试中被考察到,子串、子序列 的问题想必能排在 数一数二 的位置上。


在之前的 「动态规划」 系列文章中,我们讲到了 最长公共子序列 和 最长回文子序列 的问题,今天我们继续来探讨力扣上一个关于 子串 的问题。

3.无重复字符的最长子串

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

示例 1:

输入: s = “abcabcbb”

输出: 3

解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。

示例 2:

输入: s = “bbbbb”

输出: 1

解释: 因为无重复字符的最长子串是 “b”,所以其长度为 1。


小 tips: 要注意分清楚,子串子序列 的区别哟~

  • 子串: 必须连续
  • 子序列: 可以不连续

思路分析

这里回顾一个 重要思想 ,对于 子串和子序列 的题目,可以按如下方式进行思考:

考虑 以 i 位置为结尾 的情况下,答案如何选取

该思想在 数组求和-2 这篇文章中也有提到哦~

因此,对于本题来说:

  • 考虑若以每个位置作为结尾时,子串能够向前延伸多长,最长的子串长度就是我们要求的答案。

那么问题就进一步转化为:

  • 在给定一个结尾的字符时,应该如何向前延伸呢,延伸的长度会受到哪些因素影响呢?

稍加思考:

  1. 由于要找到最长无重复的子串,因此一定与该字符 相同的前一个字符 的位置有关。

  • 例如,假设 3 到 8 下标之间没有再出现a字符,则以 9 号下标为结尾的a字符往前延伸的距离最多只能到下标 3 处。
  1. 除了因素 1 外,也与该字符的 前一个字符 向前延伸的位置有关。

  • 同样例子,假设下标为 9a字符的前一个字符是b, 6 到 7 下标之间没有再出现b字符,则以 8 号下标为结尾的b字符往前延伸的距离最多只能到下标 6 处。
  • 进而导致了下标为 9a字符往前延伸的距离最多也只能到下标 6 处。

确定了影响最终答案的因素后,思路便豁然开朗了:

两个因素中结果较大的下标即为该位置所能扩充的最远距离。

  1. 需要解决能够找到前一个相同字符下标的方法;(使用map)
  2. 设置存储前一个字符 最远能够向前扩充的下标 变量。
  3. 取 1,2 中 较大的下标 即为该位置字符的答案。

代码

public static int lengthOfLongestSubstring(String s) {if (s == null || s.equals("")) {return 0;}char[] str = s.toCharArray();// 这里并没有直接使用 map , 与 map 功能类似// 该 map 数组中存放的是 该字符 上一次出现时 的 下标int[] map = new int[256];for (int i = 0; i < 256; i++) {map[i] = -1;}// 最新的答案(即此前最长的子串长度)int len = 0;// 前一个字符能够向前扩充的最远位置在哪int pre = -1;// 当前位置字符能够向前扩充的最远位置在哪int cur = 0;for (int i = 0; i != str.length; i++) {// 取两个因素中的最大值pre = Math.max(pre, map[str[i]]);// 此时能够扩充的最大距离cur = i - pre;// 更新答案len = Math.max(len, cur);// 更新最新该字符出现的位置map[str[i]] = i;}return len;
}

理解了本题的思想之后,上述代码也不难看懂。小伙伴们仔细思考一下哟!!!

写在最后

前面的算法文章,更新了许多 专题系列 。包括:滑动窗口、动态规划、加强堆、二叉树递归套路 等。

还没读过的小伙伴可以关注同名号,在主页中点击对应链接查看哦~

接下来的一段时间,将持续 「力扣高频题」 系列文章,想刷 力扣高频题 的小伙伴也可以关注一波哦 ~

~ 点赞 ~ 关注 ~ 星标 ~ 不迷路 ~!!!

关注
回复「ACM紫书」获取 ACM 算法书籍 ~
回复「算法导论」获取 算法导论第3版 ~

在看 + 转发

让你的小伙伴们一起来学算法吧!!

这篇关于【力扣高频题】003.无重复字符的最长子串的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1048529

相关文章

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

Java中字符编码问题的解决方法详解

《Java中字符编码问题的解决方法详解》在日常Java开发中,字符编码问题是一个非常常见却又特别容易踩坑的地方,这篇文章就带你一步一步看清楚字符编码的来龙去脉,并结合可运行的代码,看看如何在Java项... 目录前言背景:为什么会出现编码问题常见场景分析控制台输出乱码文件读写乱码数据库存取乱码解决方案统一使

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

如何通过try-catch判断数据库唯一键字段是否重复

《如何通过try-catch判断数据库唯一键字段是否重复》在MyBatis+MySQL中,通过try-catch捕获唯一约束异常可避免重复数据查询,优点是减少数据库交互、提升并发安全,缺点是异常处理开... 目录1、原理2、怎么理解“异常走的是数据库错误路径,开销比普通逻辑分支稍高”?1. 普通逻辑分支 v

nginx 负载均衡配置及如何解决重复登录问题

《nginx负载均衡配置及如何解决重复登录问题》文章详解Nginx源码安装与Docker部署,介绍四层/七层代理区别及负载均衡策略,通过ip_hash解决重复登录问题,对nginx负载均衡配置及如何... 目录一:源码安装:1.配置编译参数2.编译3.编译安装 二,四层代理和七层代理区别1.二者混合使用举例

MySQL中查找重复值的实现

《MySQL中查找重复值的实现》查找重复值是一项常见需求,比如在数据清理、数据分析、数据质量检查等场景下,我们常常需要找出表中某列或多列的重复值,具有一定的参考价值,感兴趣的可以了解一下... 目录技术背景实现步骤方法一:使用GROUP BY和HAVING子句方法二:仅返回重复值方法三:返回完整记录方法四:

XML重复查询一条Sql语句的解决方法

《XML重复查询一条Sql语句的解决方法》文章分析了XML重复查询与日志失效问题,指出因DTO缺少@Data注解导致日志无法格式化、空指针风险及参数穿透,进而引发性能灾难,解决方案为在Controll... 目录一、核心问题:从SQL重复执行到日志失效二、根因剖析:DTO断裂引发的级联故障三、解决方案:修复

SpringBoot+Redis防止接口重复提交问题

《SpringBoot+Redis防止接口重复提交问题》:本文主要介绍SpringBoot+Redis防止接口重复提交问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录前言实现思路代码示例测试总结前言在项目的使用使用过程中,经常会出现某些操作在短时间内频繁提交。例

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据