leetcode459. 重复的子字符串
Posted dengjiayue
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了leetcode459. 重复的子字符串相关的知识,希望对你有一定的参考价值。
题目描述
给定一个非空的字符串 s ,检查是否可以通过由它的一个子串重复多次构成。
示例 1:
输入: s = "abab" 输出: true 解释: 可由子串 "ab" 重复两次构成。 示例 2:
输入: s = "aba" 输出: false 示例 3:
输入: s = "abcabcabcabc" 输出: true 解释: 可由子串 "abc" 重复四次构成。 (或子串 "abcabc" 重复两次构成。)
提示:
1 <= s.length <= 104 s 由小写英文字母组成
来源:力扣(LeetCode) 链接:https://leetcode.cn/problems/repeated-substring-pattern 著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解题思路
法1
双指针法:\\
-
循环是由首字母开始的我们只需要找到字符串之后的首字母的位置开始做循环判断, -
如果中途失败,就重新做下一个首字母的循环判断,直到无法判断循环或者成功的时候返回结果
-
时间复杂度(O(n)) -
空间复杂度(O(1))
执行结果
法1
func repeatedSubstringPattern(s string) bool
length := len(s)
for i := 1; i <= length/2; i++
if length%i == 0
sub := s[:i]
repeat := length / i
isRepeated := true
for j := 1; j < repeat; j++
if s[j*i:(j+1)*i] != sub
isRepeated = false
break
if isRepeated
return true
return false
执行结果: 通过 显示详情 查看示例代码 添加备注
执行用时: 8 ms , 在所有 Go 提交中击败了 48.79% 的用户 内存消耗: 4.2 MB , 在所有 Go 提交中击败了 73.97% 的用户 通过测试用例: 129 / 129
本文由 mdnice 多平台发布
[LeetCode]剑指 Offer 48. 最长不含重复字符的子字符串
请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度。
示例 1:
输入: “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。
示例 2:
输入: “bbbbb”
输出: 1
解释: 因为无重复字符的最长子串是 “b”,所以其长度为 1。
示例 3:
输入: “pwwkew”
输出: 3
解释: 因为无重复字符的最长子串是 “wke”,所以其长度为 3。
请注意,你的答案必须是 子串 的长度,“pwke” 是一个子序列,不是子串。
提示:
- s.length <= 40000
题解一:
动态规划解析:
-
状态定义: 设动态规划列表 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,即 s[j] 左边无相同字符,则
-
返回值:max(dp) ,即全局的 “最长不重复子字符串” 的长度。
/**
* 剑指 Offer 48. 最长不含重复字符的子字符串
*/
public int lengthOfLongestSubstring(String s)
// 存储各字符最后一次出现的下标
Map<Character, Integer> map = new HashMap<>();
int res = 0;
int tmp = 0;
for (int j = 0; j < s.length(); j++)
// 获取当前字符的索引
int i = map.getOrDefault(s.charAt(j), -1);
map.put(s.charAt(j), j);
// dp[j - 1] -> dp[j]
tmp = tmp < j - i ? tmp + 1 : j - i;
// max(dp[j - 1], dp[j])
res = Math.max(res, tmp);
return res;
题解二:
/**
* 剑指 Offer 48. 最长不含重复字符的子字符串
*/
public int lengthOfLongestSubstring(String s)
int res = 0;
int tmp = 0;
for(int j = 0; j < s.length(); j++)
int i = j - 1;
// 线性查找 i
while(i >= 0 && s.charAt(i) != s.charAt(j))
i--;
// dp[j - 1] -> dp[j]
tmp = tmp < j - i ? tmp + 1 : j - i;
// max(dp[j - 1], dp[j])
res = Math.max(res, tmp);
return res;
题解三:
/**
* 剑指 Offer 48. 最长不含重复字符的子字符串
*/
public int lengthOfLongestSubstring(String s)
Map<Character, Integer> dic = new HashMap<>();
int i = -1, res = 0;
for(int j = 0; j < s.length(); j++)
if(dic.containsKey(s.charAt(j)))
// 更新左指针 i
i = Math.max(i, dic.get(s.charAt(j)));
dic.put(s.charAt(j), j);
// 更新结果
res = Math.max(res, j - i);
return res;
来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/zui-chang-bu-han-zhong-fu-zi-fu-de-zi-zi-fu-chuan-lcof
以上是关于leetcode459. 重复的子字符串的主要内容,如果未能解决你的问题,请参考以下文章
LeetCode459. 重复的子字符串,KMP+暴力双循环