当前位置: 首页 > news >正文

闯关leetcode——3.Longest Substring Without Repeating Characters

大纲

  • 题目
    • 地址
    • 内容
  • 解题
    • 代码地址

题目

地址

https://leetcode.com/problems/longest-substring-without-repeating-characters/description/

内容

Given a string s, find the length of the longest substring without repeating characters.

Example 1:

Input: s = “abcabcbb”
Output: 3
Explanation: The answer is “abc”, with the length of 3.

Example 2:

Input: s = “bbbbb”
Output: 1
Explanation: The answer is “b”, with the length of 1.

Example 3:

Input: s = “pwwkew”
Output: 3
Explanation: The answer is “wke”, with the length of 3.
Notice that the answer must be a substring, “pwke” is a subsequence and not a substring.

Constraints:

  • 0 <= s.length <= 5 * 104
  • s consists of English letters, digits, symbols and spaces.

解题

这题是要在一个只包含English letters, digits, symbols and spaces的字符串中找打最长不重复子串。
一种比较容易想到的思路是:遇到重复的字符,则从上次这个字符出现的位置的后个位置开始尝试比较。比如abcdef是当前最长不重复子串。如果后面出现的字符是b,则从下标为1的b的后一位(c)开始组成新的测试子串,即cdefb。当新的测试子串的长度大于当前最长的子串,则更新该最大长度值。
为了判断后续出现的字符是否已经存在于当前最长子串中,则可以考虑使用map<char,int>的结构来存储最长子串及其坐标信息。但是这个题目存在只包含English letters, digits, symbols and spaces的条件,这意味着每个字符都是ASCII码中的字符——它只有128个——且是连续的数字(见https://baike.baidu.com/item/ASCII/309296)。我们可以充分利用这个条件,使用一个更加紧凑高效的结构来存储这个关系。

        array<int, 128> charIndex;charIndex.fill(-1);

然后我们遍历整个字符串,将这样的关系存储起来。

for (int i = 0; i < s.size(); i++) {
……charIndex[int(s[i])] = i;
……
}

这儿有一个问题:在寻找过程中,如果最长子串发生了变化,则被剔除出的字符串是否应该从charIndex中剔除?比如上例abcdef是最长子串时,charIndex包含这些字符以及它们的位置。如果在f后面新出现的字符是b,则要开始新探索的子串是cdefb,这个时候字符a是否需要从charIndex中剔除?因为它可能会影响到后续再次出现a时的判断。实际我们通过新探索的子串开头第一个字符位置来判断新的重复是否需要处理。在这个例子中,第一个a(绿色)位于新子串cdefb起始位置(箭头处)的左侧,即使它存在charIndex中,也不会影响后续的探索,即我们可将新的探索子串定义为cdefba。假设下一个出现的是字符d,由于上一次出现的d位于子串起始位置(箭头处)的右侧,所以需要重新探索新的子串
在这里插入图片描述
经过分析,我们知道子串起始位置也是需要记录,以用于辅助判断是否要使用新子串。
由于该题只是要输出最长子串长度,并不要求输出最长子串的内容,所以我们并不用记录子串的具体顺序,只要比较并保留最长的一条串的长度即可。
最长串的比较maxLen = max(maxLen, i - start)可以优化到串的起始位置发生变化的时候进行。这样可以将对比的次数降到最低。

#include <string>
#include <array>
using namespace std;class Solution {
public:int lengthOfLongestSubstring(string s) {array<int, 128> charIndex;charIndex.fill(-1);int start = 0;int maxLen = 0;for (int i = 0; i < s.size(); i++) {if (charIndex[int(s[i])] >= start) {maxLen = max(maxLen, i - start);start = charIndex[int(s[i])] + 1;}charIndex[int(s[i])] = i;}if (s.size() - start > maxLen) {maxLen = s.size() - start;}return maxLen;}
};

上述代码的一个边界问题是:最长子串的尾部正好是原始串的尾部。这样我们必须在遍历结束后,考虑这种情况下最长子串的可能性。

在这里插入图片描述

代码地址

https://github.com/f304646673/leetcode/tree/main/3-Longest-Substring-Without-Repeating-Characters

http://www.lryc.cn/news/433458.html

相关文章:

  • Android Radio2.0——公告注册及监听(三)
  • 【C++】类和对象(三)再探构造函数|static成员函数|友元函数|内部类|匿名对象|对象拷贝时的编译优化
  • 2024中国算力大会 2024 China Computational Power Conference
  • jEasyUI 扩展行显示细节
  • YOLOv8+Deepsort+PyQt+GUI 语义分割+目标检测+姿态识别 三者合一(集成于一套系统)综合视觉分析系统
  • 机器学习无监督学习
  • windows10-VMware17-Ubuntu-22.04-海康2K摄像头兼容问题,求解(已解决)
  • 【系统架构设计师】解释器模式
  • Hive原理剖析
  • 在 Ubuntu 上查看重复文件
  • docker容器高效连接 Redis 的方式
  • 手撕Python之生成器、装饰器、异常
  • LabVIEW步进电机控制方式
  • vllm源码解析(五):LLM模型推理
  • 数学建模笔记——熵权法(客观赋权法)
  • XGBoost算法-确定树的结构
  • concurrentHashMap线程安全实现的原理
  • 域名证书,泛域名证书,sni
  • Pytest夹具autouse参数使用。True表示会自动在测试中使用,而无需显式指定
  • Linux:归档及压缩
  • jenkins 安装
  • mysql学习教程,从入门到精通,MySQL 删除数据库教程(6)
  • C语言:刷题日志(2)
  • 微带结环行器仿真分析+HFSS工程文件
  • 怎么仿同款小程序的开发制作方法介绍
  • 音视频入门基础:WAV专题(10)——FFmpeg源码中计算WAV音频文件每个packet的pts、dts的实现
  • 0.91寸OLED屏幕大小的音频频谱,炫酷
  • 6. LinkedList与链表
  • Statcounter Global Stats 提供全球统计数据信息
  • Linux kernel中的dts dtsi dtb dtc dtb.img dtbo.img