#1传送门

无重复字符的最长子串

给定一个字符串 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 由英文字母、数字、符号和空格组成

思路

容易想到这里需要使用滑动窗口的思想来解决问题

我们可以定义两个指针,left和right

left一开始指向0,right则放入for循环中向前不断遍历字符串s

这里还需要维护一个哈希表(选择map,因为字符类型不止有26个字母)

使用当前遍历值(字符)在表中查询,如果没出现过,就把当前字符放入哈希表中

注意!!!!!!

这里有可能我们会下意识的将哈希表的键值结构定义为:"遍历字符--字符出现次数"

在本题中,这是错误的,或者说是不合适的

因为我们需要找到重复字符第一次出现的位置,如以字符出现次数为值的话,无法实现这一目的(字符出现位置可以帮助我们确定哪个字符是第一个不重复的字符

举个例子,假设有字符串 "abccba",如果我们以字符出现次数为值来构建哈希表,那么哈希表的值应该为 {'a': 2, 'b': 2, 'c': 2},这些字符的出现次数都是相同的。如果我们想要找到第一个不重复的字符,就需要进一步遍历原字符串,找到第一个出现次数为1的字符。但是,如果我们以字符出现位置为值来构建哈希表,那么哈希表的值应该为 {'a': 0, 'b': 1, 'c': 2},我们可以在遍历字符串的过程中实时更新哈希表,并找到第一个出现位置最小的字符。

因此,在这个问题中,以字符为键,以字符出现位置为值("遍历字符--字符首次出现位置")是更加合适的选择。

继续

如果当前遍历值在表中出现过,那么我们在哈希表中获取该字符对应的值,即其在s中第一次出现的位置

将左指针移动到该位置(为什么?),并且加1跳过该位置,目的是剔除重复值

这里是本题的第二个坑

在写的时候,第一版中我移动left时用了left++,但是这样是错的,不能简单地使用left++来移动左指针,因为有可能之前的某个字符已经出现过,那么我们需要更新左指针的位置,使得新的左指针位置可以舍弃之前出现过的字符,从而保证当前的子串不重复。

这其实还涉及到我们如何定义“重复出现”这件事情

因为在设计hash表时,保存的是当前字符的索引,因此,我们仅仅在hash中查询到某个字符还不够,原因如下:

如果当前字符s[right]已经在哈希表中存在,则说明该字符之前出现过,我们需要更新左指针的位置,使其跳过该重复字符,即左指针left的新位置应该是(hash[s[right]]+1)。但是,在更新左指针的同时,还需要确保新的左指针位置大于等于上一个子串的起始位置left,因为我们要寻找最长的无重复子串,而不仅仅是子串长度

因此,hash.find(s[right]) != hash.end() && hash[s[right]] >= left 确保了字符s[right]曾经出现过(也就是该字符在哈希表中有对应的索引),并且其最后一次出现的下标大于等于当前子串的起始位置left,才会更新左指针的位置。

如果只写 hash.find(s[right]) != hash.end() 的话,可能会把之前的那些已经被跳过的字符再次算进去,从而导致错误的结果。

例如:"abcabcbb"

如果当前左指针指向的是第一个b,右指针指向第二个a,说明我们已经判断a重复出现,并把左指针移动到了hash[s[right]]+1

此时我们并没有删除hash表中关于a第一次出现的位置信息,因此下一次如果还遇到a,我们不能让左指针移动到第一次a的位置

所以需要加上限定条件,即hash[s[right]] >= left来保证左指针的正常跳转

然后,更新当前字符在哈希表中的位置值,即将当前字符位置设置为第一次出现位置

代码

关键点:哈希表键值对设计(采用"遍历字符--字符首次出现位置")、left指针的移动方式

步骤:

1、定义一个哈希表,键为字符,值为字符出现的index

2、移动右指针遍历字符串s,在哈希表中查询当前遍历字符

​ 如果有重复,同时判断该重复值位置是否大于left指针位置(因为如果出现重复值,left指针所值的应该是上一次该值出现的位置),大于则将left移动到当前字符位置,并加1跳过当前位置

​ 如果无重复,先不处理

3、不管有无重复都对当前遍历字符在哈希表中的值(即位置索引)进行更新(同时也处理了无重复的情况)

4、左右指针作差与最大长度变量比较,并更新该变量

5、返回最大长度变量

class Solution {
public:
int lengthOfLongestSubstring(string s) {
unordered_map<char, int> hash;//创建hash表
int left = 0;//左指针
int maxLen = 0; for(int right = 0; right < s.size(); ++right){//遍历字符串s
if(hash.find(s[right]) != hash.end() && left <= hash[s[right]]){//若当前字符在哈希表中存在
left = hash[s[right]] + 1;//获取当前字符在哈希表中对应的值,该值为字符在s中的索引,将左指针移动到此处
//即若当前字符在哈希表中存在,我们需要将left指针指到重复字符s[right]上次出现的位置,然后加1来跳过它
}
//对应情况:1、当前字符在哈希表中不存在/2、跳转left指针至重复值第一次出现位置之后,更新当前字符在哈希表中的位置信息
hash[s[right]] = right;//添加字符到哈希表/更新信息
if(right - left + 1 > maxLen) maxLen = right - left + 1;//更新当前最大长度
}
return maxLen;
}
};

【LeetCode滑动窗口专题#2】无重复字符的最长子串的更多相关文章

  1. LeetCode 第 3 题:无重复字符的最长子串(滑动窗口)

    LeetCode 第 3 题:无重复字符的最长子串 (滑动窗口) 方法:滑动窗口 滑动窗口模板问题:右指针先走,满足了一定条件以后,左指针向前走,直到不满足条件. 特点:左右指针的方向是一致的,并且是 ...

  2. leetcode刷题笔记-3. 无重复字符的最长子串(java实现)

    题目描述 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例 1: 输入: "abcabcbb"输出: 3 解释: 因为无重复字符的最长子串是 "ab ...

  3. LeetCode随缘刷题之无重复字符的最长子串

    欢迎评论区交流. package leetcode.day_12_04; /** * 给定一个字符串 s ,请你找出其中不含有重复字符的最长子串的长度. * <p> * 示例1: * &l ...

  4. LeetCode(3):无重复字符的最长子串

    Medium! 题目描述: 给定一个字符串,找出不含有重复字符的 最长子串 的长度. 示例: 给定 "abcabcbb" ,没有重复字符的最长子串是 "abc" ...

  5. 【leetcode】 算法题3 无重复字符的最长子串

      问题      给定一个字符串,找出不含有重复字符的最长子串的长度. 示例: 给定 "abcabcbb" ,没有重复字符的最长子串是 "abc" ,那么长度 ...

  6. leetcode的Hot100系列--3. 无重复字符的最长子串--滑动窗口

    可以先想下这两个问题: 1.怎样使用滑动窗口? 2.如何快速的解决字符查重问题? 滑动窗口 可以想象一下有两个指针,一个叫begin,一个叫now 这两个指针就指定了当前正在比较无重复的字符串,当再往 ...

  7. 【LeetCode】无重复字符的最长子串【滑动窗口法】

    给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例 1: 输入: "abcabcbb"输出: 3 解释: 因为无重复字符的最长子串是 "abc&quo ...

  8. [LeetCode]3. 无重复字符的最长子串(滑动窗口)

    题目 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例 1: 输入: "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 "abc ...

  9. 滑动窗口经典题 leetcode 3. 无重复字符的最长子串

    题目 解题思路 题目要求找出给定字符串中不含有重复字符的最长子串的长度.这是一个典型的滑动窗口的题目,可以通过滑动窗口去解答. 滑动窗口 具体操作如下图示:找到一个子串 s[left...right] ...

  10. Leetcode(三)无重复字符的最长子串

    3. 无重复字符的最长子串 题目描述 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例: 输入: "abcabcbb" 输出: 3 解释: 因为无重复字符的最 ...

随机推荐

  1. Mybatis Plus根据某字段特定值排序

    需求 背景:一个审核流程.审核人等级分为市级和省级,管理员升级字段adminlevel,字段含义:1省级,2市级.审核字段audit为int字段,字段含义:1待市级审核,2待省级审核,3通过审核. 需 ...

  2. 二进制安装Kubernetes(k8s) v1.26.0 IPv4/IPv6双栈

    二进制安装Kubernetes(k8s) v1.26.0 IPv4/IPv6双栈 https://github.com/cby-chen/Kubernetes 开源不易,帮忙点个star,谢谢了 介绍 ...

  3. flask-sqlalchemy入门

    Flask-SQLAlchemy 是一个为 Flask 应用增加 SQLAlchemy 支持的扩展.它致力于简化在 Flask 中 SQLAlchemy 的使用.SQLAlchemy 是目前pytho ...

  4. [MAUI 项目实战] 手势控制音乐播放器(一): 概述与架构

    这是一篇系列博文.请关注我,学习更多.NET MAUI开发知识! [MAUI 项目实战] 手势控制音乐播放器(一): 概述与架构 [MAUI 项目实战] 手势控制音乐播放器(二): 手势交互 [MAU ...

  5. openGauss Datakit安装部署

    一.问题描述:目前找不到任何关于opengauuss Datakit安装部署的文档,自己来尝试踩坑. DataKit是一个以资源(物理机,数据库)为底座的开发运维工具,将上层的开发运维工具插件化,各插 ...

  6. 详解C++中的extern与static关键字

    本章通过问答方式明晰两个关键字及其作用. Q1:对于int x:,不加extern关键字他就是个未赋初值的定义,但是如果加了static或者extern都可以表示这仅是一个声明吗? A:不是的,具体情 ...

  7. 自定义Feign拦截器

    简介 Feign的拦截器RequestInterceptor SpringCloud的微服务使用Feign进行服务间调用的时候可以使用RequestInterceptor统一拦截请求来完成设置head ...

  8. 玩转云端 | 算力基础设施升级,看天翼云紫金DPU显身手!

    数字时代下,算力成为新的核心生产力,传统以CPU为核心的架构难以满足新场景下快速增长的算力需求,具备软硬加速能力的DPU得以出现并快速发展.天翼云凭借领先的技术和丰富的应用实践自研紫金DPU,打造为云 ...

  9. 记一次 Windows10 内存压缩模块 崩溃分析

    一:背景 1. 讲故事 在给各位朋友免费分析 .NET程序 各种故障的同时,往往也会收到各种其他类型的dump,比如:Windows 崩溃,C++ 崩溃,Mono 崩溃,真的是啥都有,由于基础知识的相 ...

  10. vue-cli3构建和发布 实现分环境打包步骤(给不同的环境配置相对应的打包命令)

    https://panjiachen.github.io/vue-element-admin-site/zh/guide/essentials/deploy.html#%E6%9E%84%E5%BB% ...