题目信息
-
题目:剑指 Offer 48. 最长不含重复字符的子字符串
-
时间: 2020-08-24
-
题目链接:Leetcode
-
tag: 动态规划 哈希表
-
难易程度:中等
-
题目描述:
请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度。
示例1:
1 | 输入: "abcabcbb" |
示例2:
1 | 输入: "bbbbb" |
示例3:
1 | 输入: "pwwkew" |
注意
1 | 1. s.length <= 40000 |
解题思路
本题难点
长度为 N 的字符串共有 (1+N)N/2 个子字符串(复杂度为 O(N^2 ) ),判断长度为 N 的字符串是否有重复字符的复杂度为 O(N) ,使用暴力法解决的复杂度为 O(N^3) 。
具体思路
考虑使用动态规划降低时间复杂度。
- 状态定义:设动态规划列表 dp ,dp[j] 代表以字符 s[j 为结尾的 “最长不重复子字符串” 的长度。
- 转移方程:固定右边界 j ,设字符 s[j] 左边距离最近的相同字符为 s[i] ,即 s[i]=s[j] 。
- 当 i<0时,即 s[j] 左边无相同字符,则 dp[j]=dp[j−1]+1 ;
- 当 dp[j−1] < j−i 时,说明字符 s[i] 在子字符串 dp[j−1] 区间之外 ,则 dp[j]=dp[j−1]+1 ;
- 当 dp[j−1] > j−i 时,说明字符 s[i] 在子字符串 dp[j−1] 区间之中 ,则 dp[j] 的左边界由 s[i] 决定,即 dp[j]=j−i ;
注意:当 i<0 时,由于 dp[j−1]≤j 恒成立,因而 dp[j−1]<j−i 恒成立,因此分支 1. 和 2. 可被合并。
代码
1 | class Solution { |
复杂度分析:
- 时间复杂度 O(N) :其中 N为字符串长度,动态规划需遍历计算 dp列表。
- 空间复杂度 O(1) : 字符的 ASCII 码范围为 0 ~ 127 ,哈希表 dic 最多使用 O(128)=O(1) 大小的额外空间。
其他优秀解答
解题思路
双指针 + 哈希表
哈希表 dic统计: 指针 j遍历字符 s,哈希表统计字符 s[j]最后一次出现的索引 。
更新左指针 i : 根据上轮左指针 i 和 dic[s[j]] ,每轮更新左边界 i ,保证区间 [i+1,j] 内无重复字符且最大
更新结果 res : 取上轮 res 和本轮双指针区间 [i+1,j] 的宽度(即 j−i )中的最大值。
代码
1 | class Solution { |