代码随想录训练营day09|459.重复的子字符串


前言

代码随想录算法训练营day09


一、459.重复的子字符串

1.题目

给定一个非空的字符串 s ,检查是否可以通过由它的一个子串重复多次构成。

示例 1:

输入: s = "abab" 输出: true 解释: 可由子串 "ab" 重复两次构成。

示例 2:

输入: s = "aba" 输出: false

示例 3:

输入: s = "abcabcabcabc" 输出: true 解释: 可由子串 "abc" 重复四次构成。 (或子串 "abcabc" 重复两次构成。)

提示:

复制代码

matlab

复制代码

1 <= s.length <= 104 s 由小写英文字母组成

来源:力扣(LeetCode) 链接:leetcode.cn/problems/re...

2.解题思路

  1. 方法一:枚举

如果一个长度为 nn 的字符串 ss 可以由它的一个长度为 n′n′ 的子串 s′s′ 重复多次构成,那么:

复制代码

css

复制代码

nn 一定是 n′n′ 的倍数; s′s′ 一定是 ss 的前缀; 对于任意的 i∈[n′,n)i∈[n′,n),有 s[i]=s[i−n′]s[i]=s[i−n′]。

也就是说,ss 中长度为 n′n′ 的前缀就是 s′s′,并且在这之后的每一个位置上的字符 s[i]s[i],都需要与它之前的第 n′n′ 个字符 s[i−n′]s[i−n′] 相同。

因此,我们可以从小到大枚举 n′n′,并对字符串 ss 进行遍历,进行上述的判断。注意到一个小优化是,因为子串至少需要重复一次,所以 n′n′ 不会大于 nn 的一半,我们只需要在 [1,n2][1,2n​] 的范围内枚举 n′n′ 即可。

  1. 方法二:字符串匹配

我们可以把字符串 ss 写成

复制代码

复制代码

s′s′⋯s′s′s′s′⋯s′s′

的形式,总计 nn′n′n​ 个 s′s′。但我们如何在不枚举 n′n′ 的情况下,判断 ss 是否能写成上述的形式呢?

如果我们移除字符串 ss 的前 n′n′ 个字符(即一个完整的 s′s′),再将这些字符保持顺序添加到剩余字符串的末尾,那么得到的字符串仍然是 ss。由于 1≤n′<n1≤n′<n,那么如果将两个 ss 连在一起,并移除第一个和最后一个字符,那么得到的字符串一定包含 ss,即 ss 是它的一个子串。

因此我们可以考虑这种方法:我们将两个 ss 连在一起,并移除第一个和最后一个字符。如果 ss 是该字符串的子串,那么 ss 就满足题目要求。

注意到我们证明的是如果 ss 满足题目要求,那么 ss 有这样的性质,而我们使用的方法却是如果 ss 有这样的性质,那么 ss 满足题目要求。因此,只证明了充分性是远远不够的,我们还需要证明必要性。

证明需要使用一些同余运算的小技巧,可以见方法三之后的「正确性证明」部分。这里先假设我们已经完成了证明,这样就可以使用非常简短的代码完成本题。在下面的代码中,我们可以从位置 11 开始查询,并希望查询结果不为位置 nn,这与移除字符串的第一个和最后一个字符是等价的。

3.代码实现

方法一:枚举

复制代码

java

复制代码

class Solution { public boolean repeatedSubstringPattern(String s) { int n = s.length(); for (int i = 1; i * 2 <= n; ++i) { if (n % i == 0) { boolean match = true; for (int j = i; j < n; ++j) { if (s.charAt(j) != s.charAt(j - i)) { match = false; break; } } if (match) { return true; } } } return false; } }

方法二:字符串匹配

复制代码

java

复制代码

class Solution { public boolean repeatedSubstringPattern(String s) { return (s + s).indexOf(s, 1) != s.length(); } }

相关推荐
徐小夕@趣谈前端5 分钟前
Web文档的“Office时刻“:jitword共建版2.0发布!让浏览器变成本地生产力
前端·数据结构·vue.js·算法·开源·编辑器·es6
问好眼9 分钟前
【信息学奥赛一本通】1275:【例9.19】乘积最大
c++·算法·动态规划·信息学奥赛
Daydream.V27 分钟前
逻辑回归实例问题解决(LogisticRegression)
算法·机器学习·逻辑回归
代码无bug抓狂人28 分钟前
C语言之表达式括号匹配
c语言·开发语言·算法
不穿格子的程序员35 分钟前
从零开始写算法——普通数组篇:缺失的第一个正数
算法·leetcode·哈希算法
Nebula_g42 分钟前
线程进阶: 无人机自动防空平台开发教程(更新)
java·开发语言·数据结构·学习·算法·无人机
rit84324991 小时前
基于MATLAB的环境障碍模型构建与蚁群算法路径规划实现
开发语言·算法·matlab
hoiii1871 小时前
MATLAB SGM(半全局匹配)算法实现
前端·算法·matlab
独自破碎E1 小时前
大整数哈希
算法·哈希算法
纤纡.1 小时前
逻辑回归实战进阶:交叉验证与采样技术破解数据痛点(二)
算法·机器学习·逻辑回归