POJ-1743 Musical Theme 字符串问题 不重叠最长重复子串
题目链接:https://cn.vjudge.net/problem/POJ-1743
题意
给一串整数,问最长不可重叠最长重复子串有多长
注意这里匹配的意思是匹配串的所有元素可以减去或者加上某个值
例: 34 30 26 22 18 82 78 74 70 66
后5个整数的串可以匹配前5个数
思路
LCP问题(最长公共前缀)
两个思路
- 后缀数组
对height数组二分长度,找到height大于len且两字符串起点差值大于len的情况下,len最大值 - 哈希+二分
二分长度,哈希值比较字符串找到当前len下符合要求子串即可
这题我用的第二个思路
首先差分一下,然后二分长度,用哈希的方法比较两个字符串O(1)
总复杂度O(nlog)(初始化O(n)+二分O(1nlogn))
刚从蓝书看来的,然而本题对hash表要求很高(能够处理unsigned long long大小键值)
那个简洁的HashMap还是要改改
具体写下好了
首先构建一个H数组,具体的
\]
对一个以i为起点,len为长度的字符串,他的Hash值:
\]
然后O(n)的字符串比较复杂度降到O(1)
这里可能会撞hash,所以保险一点咱还得再直接判断是否相同
提交过程
| WA*n | |
| TLE*n | |
| AC |
代码
#include <cstdio>
#include <cstring>
const int maxn=2e4+20;
const int HASH = 10007;
const int MAXN = 20010;
const unsigned long long hashBase=13331;
struct HASHMAP{
int head[HASH],next[MAXN],size;
unsigned long long state[MAXN];
int f[MAXN];
void init(){
size = 0;
memset(head,-1,sizeof(head));
}
int insert(unsigned long long val,int _id){
int h = val%HASH;
for(int i = head[h]; i != -1;i = next[i])
if(val == state[i]) return f[i];
f[size] = _id;
state[size] = val;
next[size] = head[h];
head[h] = size++;
return f[size-1];
}
}hash;
unsigned long long hashBasePow[maxn], H[maxn];
int str[maxn], n;
bool judge(int len){
hash.init();// hash.clear();
for(int i=0; i<n-len; i++){
unsigned long long key=H[i]-H[i+len]*hashBasePow[len];
if (hash.insert(key, i)<i-len) return true;
}return false;
}
int solve(void){
int ans=0;
int l=4, r=n-1;
while(l<=r){
int mid=l+(r-l)/2;
if(judge(mid)) ans=mid, l=mid+1;
else r=mid-1;
}
if(ans<4) ans=-1;
return ans+1;
}
int main(){
hashBasePow[0]=1;
for(int i=1; i<maxn; i++)
hashBasePow[i]=hashBasePow[i-1]*hashBase;
while(scanf("%d", &n)==1 && n){
for (int i=0; i<n; i++) scanf("%d", &str[i]);
for (int i=0; i<n-1; i++) str[i]-=str[i+1];
H[n-1]=str[n-1];
for (int i=n-2; i>=0; i--)
H[i]=H[i+1]*hashBase+str[i];
printf("%d\n", solve());
}
return 0;
}
| Time | Memory | Length | Lang | Submitted |
|---|---|---|---|---|
| 547ms | 1080kB | 1557 | G++ | 2018-08-03 15:26:17 |
POJ-1743 Musical Theme 字符串问题 不重叠最长重复子串的更多相关文章
- POJ 1743 Musical Theme 后缀数组 不可重叠最长反复子串
二分长度k 长度大于等于k的分成一组 每组sa最大的和最小的距离大于k 说明可行 #include <cstdio> #include <cstring> #include & ...
- [Poj1743] [后缀数组论文例题] Musical Theme [后缀数组不可重叠最长重复子串]
利用后缀数组,先对读入整数处理str[i]=str[i+1]-str[i]+90这样可以避免负数,计算Height数组,二分答案,如果某处H<lim则将H数组分开,最终分成若干块,判断每块中是否 ...
- POJ 1743 Musical Theme (后缀数组,求最长不重叠重复子串)(转)
永恒的大牛,kuangbin,膜拜一下,Orz 链接:http://www.cnblogs.com/kuangbin/archive/2013/04/23/3039313.html Musical T ...
- POJ 1743 Musical Theme(不可重叠最长重复子串)
题目链接:http://poj.org/problem?id=1743 题意:有N(1 <= N <=20000)个音符的序列来表示一首乐曲,每个音符都是1..88范围内的整数,现在要找一 ...
- poj 1743 Musical Theme(最长重复子串 后缀数组)
poj 1743 Musical Theme(最长重复子串 后缀数组) 有N(1 <= N <=20000)个音符的序列来表示一首乐曲,每个音符都是1..88范围内的整数,现在要找一个重复 ...
- Poj 1743 Musical Theme (后缀数组+二分)
题目链接: Poj 1743 Musical Theme 题目描述: 给出一串数字(数字区间在[1,88]),要在这串数字中找出一个主题,满足: 1:主题长度大于等于5. 2:主题在文本串中重复出现 ...
- POJ - 1743 Musical Theme (后缀数组)
题目链接:POJ - 1743 (不可重叠最长子串) 题意:有N(1<=N<=20000)个音符的序列来表示一首乐曲,每个音符都是1..88范围内的整数,现在要找一个重复的子串,它需要 ...
- POJ 1743 Musical Theme(后缀数组)
题意:有n个数值,算出相邻两个值的差值,此时有n-1个值的序列,把这序列当做字符串的话,求最长重复子串,且这两个子串不能重叠. 分析:后缀数组解决.先二分答案,把题目变成判定性问题:判断是否存在两个长 ...
- poj 1743 男人八题之后缀数组求最长不可重叠最长重复子串
Musical Theme Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 14874 Accepted: 5118 De ...
随机推荐
- 阿里云数据库MySQL版快速上手!
MySQL是全球最受欢迎的开源数据库,其在各Web应用中均有广泛部署.阿里云数据库MySQL版基于Alibaba的MySQL源码分支,经过双11高并发.大数据量的考验,拥有优良的性能和吞吐量.除此之外 ...
- 关于使用css伪类实现小图标
效果: .person_use>span{ display:block; width:0; height:0; border-width:10px; border-style:solid; bo ...
- 清空chrome浏览器缓存
缓存是一个很深奥的东西,虽然查了半天,还是没有搞清楚,希望以后可以遇到前端大神,可以给一个傻瓜化的通俗易懂的解释 已经上线的,后续有迭代的软件,迭代的版本不应该手动清除缓存了,因为太麻烦,对客户来说不 ...
- 使用sourceMap文件定位小程序错误信息
sourceMap是什么 在前端开发过程中代码难免会有错误,即便是再小心,也有可能出现 Cannot read property 'xxx' of null 这样的低级失误,debug自然是家常便饭. ...
- 紫书 习题7-13 UVa 817(dfs+栈求表达式的值)
题目链接 点击打开链接 这道题分为两个部分, 一用搜索枚举每种可能, 二计算表达式的值, 有挺多细节需要注意 特别注意我的代码中在计算表达式的值中用到了一个!(代码枚举中的!表示不加符号, 我现在说 ...
- ubuntu/wireshark --Lua: Error during loading: [string "/usr/share/wireshark/init.lua"]:45问题解决
错误如下: 解决方案:修改init.lua 直接运行wireshark的话会报错: Lua: Error during loading:[string "/usr/share/wiresha ...
- 数组实例的 entries(),keys() 和 values()
数组实例的 entries(),keys() 和 values() entries(),keys()和values(),用于遍历数组.它们都返回一个遍历器对象,可以用for...of循环进行遍历,唯一 ...
- DML语句(添加、更新和删除记录)
a.添加记录(一次插入一行记录) insert into 表名(字段名,字段名...) values (字段值,字段值...) insert into person ...
- Java String.replaceAll()方法
声明 以下是java.lang.String.replaceAll()方法的声明 public String replaceAll(String regex, String replacement) ...
- leetcode第一刷_Subsets II
要求子集,有很现成的方法.N个数.子集的个数是2^N.每一个元素都有在集合中和不在集合中两种状态,这些状态用[0,pow(2,N)]中每一个数来穷举,假设这个数中的第i位为1,说明当前集合中包括源数组 ...