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

转码刷 LeetCode 笔记[1]:3.无重复字符的最长子串(python)

题目描述

image

初次错解

看 B 站视频后,了解到“滑动窗口”思想,遂自己动手尝试

class Solution:def lengthOfLongestSubstring(self, s: str) -> int:left = 0lenth = 0max_lenth = 0s = list(s)str = []for char in s:right = 0if char not in str:str.append(char)right += 1lenth = len(str)if lenth > max_lenth:max_lenth = lenthelse:str.remove(char)left += 1return max_lenth

答案分析

输入"abcabcbb",输出的是5,正确结果是3
输入"pwwkew",输出4,正确结果是3

第二次的解法

下面是 AI 根据我的解法修正的答案:

class Solution:def lengthOfLongestSubstring(self, s: str) -> int:left = 0lenth = 0max_lenth = 0seen = set()for right in range(len(s)):while s[right] in seen:seen.remove(s[left])left += 1seen.add(s[right])lenth = right - left + 1if lenth > max_lenth:max_lenth = lenthreturn max_lenth

我错哪了呢?

没有真正维护滑动窗口的左边界,只是 left += 1,但 str 中并没有正确地移除所有在左边界左边的字符;
remove(char) 只会移除第一个匹配项,这会导致窗口状态不一致;
也没有用 left 来限制窗口范围,只是把它当成一个计数器。

观察我的答案和修正之后的答案,产生了几个问题:

  1. seen = set()为什么用set不用list,二者有什么区别?
  2. 为什么seen.remove(s[left])能移除所有在左边界左边的字符,而str.remove(char)不能?
  3. while s[right] in seen: 改成 for s[right] in seen: 是不是没有区别?
  4. 子串长度为什么用right - left + 1计算,而不用len(seen)计算?
  5. seen = set()seen={} 有什么区别?

下面来逐一解答

1. seen = set()为什么用 set 不用 list,二者有什么区别?

特性setlist
查找速度O(1)(哈希表实现)O(n)(线性扫描)
是否允许重复❌ 不允许重复元素✅ 允许重复元素
是否保持顺序❌ 不保持插入顺序(Python 3.7+ 中 set 会保留插入顺序,但不依赖它)✅ 保持插入顺序

2. 为什么seen.remove(s[left])能移除所有在左边界左边的字符,而str.remove(char)不能?

str.remove(char)

这行代码的意思是:从 list 中删除第一个等于 char 的元素;它不会删除所有在左边界左边的字符;它只是把第一次出现的那个字符删掉;所以窗口状态就乱了,并没有真正“滑动”窗口。

while s[right] in seen:seen.remove(s[left])left += 1

通过 while 循环,可以只要出现重复字符,就删掉一次该字符,并实现左边界右移,直到窗口中没有重复子串为止。

关于循环,进而又产生了新的问题

3. while s[right] in seen: 改成 for s[right] in seen: 是不是没有区别?

有区别!

语法层面的错误

for s[right] in seen: 试图把 seen 中的每个元素赋值给 s[right],也就是试图修改原字符串 s 的内容 —— 这是不允许的(字符串是不可变的),会报错:
TypeError: ‘str’ object does not support item assignment

逻辑层面的错误

for 循环会遍历 set 中的每个元素,而不是只要重复就一直删。

4. 子串长度为什么用right - left + 1计算,而不用len(seen)计算?

right - left + 1 是窗口的实际长度,也就是从 left 到 right 之间的子串长度
len(seen) 是不重复的字符数量,当把题目改为“允许最多两个重复”时,len(seen)输出的结果就不是子串长度了
只是在这道题中,这两个值相等。

5. seen = set()seen={} 有什么区别?

纯纯的基础不牢!刷题刷懵了!

seen = set()是创建一个空集合
seen={}是创建一个空字典

写法类型是否可哈希去重
set()set✅ 是
{}dict❌ 否
http://www.dtcms.com/a/307306.html

相关文章:

  • 一对一交友小程序 / APP 系统架构分析
  • n8n为什么建议在数组的每个item中添加json键?
  • python的异步、并发开发
  • 聊下多线程查询数据库
  • YOLO---01目标检测基础
  • C++从入门到起飞之——智能指针!
  • day 40 打卡-装饰器
  • Vulnhub Thales靶机复现详解
  • 02 基于sklearn的机械学习-KNN算法、模型选择与调优(交叉验证、朴素贝叶斯算法、拉普拉斯平滑)、决策树(信息增益、基尼指数)、随机森林
  • 【JEECG】JVxeTable表格拖拽排序功能
  • C语言:逆序输出0到9的数组元素
  • LeetCode Hot 100 搜索旋转排序数组
  • 腾讯云市场排名
  • 借助 Wisdom SSH 的 AI 助手构建 Linux 开发环境
  • 2419.按位与最大的最长子数组
  • duiLib 自定义资源目录
  • 限流算法详解:固定窗口、滑动窗口、令牌桶与漏桶算法全面对比
  • P1036 [NOIP 2002 普及组] 选数
  • 结合C++红黑树与AI人工智能的应用
  • Linux 系统日志管理与时钟同步实用指南
  • TCP和UDP编程的主要区别
  • 当人生低谷无人帮助时,如何独自奏响人生乐章
  • Linux系统编程Day1-- Linux系统的概念,主要内容
  • 查看遥控器6通道(以及其他通道)的实际PWM值
  • 洛谷 P1601 A+B Problem(高精)普及-
  • Datawhale AI夏令营 大模型技术task3 稍稍提分
  • 密码学安全性简介
  • LLM—— 基于 MCP 协议(Stdio 模式)的工具调用实践
  • 从一开始的网络攻防(十三):WAF入门到上手
  • 疏老师-python训练营-Day30模块和库的导入