POJ1743 Musical Theme [后缀数组]
Time Limit: 1000MS | Memory Limit: 30000K | |
Total Submissions: 27539 | Accepted: 9290 |
Description
Many composers structure their music around a repeating &qout;theme&qout;, which, being a subsequence of an entire melody, is a sequence of integers in our representation. A subsequence of a melody is a theme if it:
- is at least five notes long
- appears (potentially transposed -- see below) again somewhere else in the piece of music
- is disjoint from (i.e., non-overlapping with) at least one of its other appearance(s)
Transposed means that a constant positive or negative value is added to every note value in the theme subsequence.
Given a melody, compute the length (number of notes) of the longest theme.
One second time limit for this problem's solutions!
Input
The last test case is followed by one zero.
Output
Sample Input
- 30
- 25 27 30 34 39 45 52 60 69 79 69 60 52 45 39 34 30 26 22 18
- 82 78 74 70 66 67 64 60 65 80
- 0
Sample Output
- 5
Hint
Source
【2017-02-06】
除了分组还有一种做法,从大到小枚举L然后合并 维护并查集内mx和mn看看是不是>L(注意并查集用的编号是排序后的排名)
为什么>L 因为 1 2 3-->1 1 两个其实是重合的
- #include <iostream>
- #include <cstdio>
- #include <cstring>
- using namespace std;
- const int N=2e4+;
- inline int read(){
- char c=getchar();int x=,f=;
- while(c<''||c>''){if(c=='-')f=-; c=getchar();}
- while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
- return x*f;
- }
- int n,m,k;
- int s[N];
- int sa[N],c[N],t1[N],t2[N],height[N],rnk[N];
- void getHeight(){
- int k=;
- for(int i=;i<=n;i++) rnk[sa[i]]=i;
- for(int i=;i<=n;i++){
- if(k) k--;
- if(rnk[i]==) continue;
- int j=sa[rnk[i]-];
- while(i+k<=n&&j+k<=n&&s[i+k]==s[j+k]) k++;
- height[rnk[i]]=k;
- }
- }
- inline bool cmp(int *r,int a,int b,int j){
- return a+j<=n&&b+j<=n&&r[a]==r[b]&&r[a+j]==r[b+j];
- }
- void getSA(){
- m=;
- int *r=t1,*k=t2;
- for(int i=;i<=m;i++) c[i]=;
- for(int i=;i<=n;i++) c[r[i]=s[i]]++;
- for(int i=;i<=m;i++) c[i]+=c[i-];
- for(int i=n;i>=;i--) sa[c[r[i]]--]=i;
- for(int j=;j<=n;j<<=){
- int p=;
- for(int i=n-j+;i<=n;i++) k[++p]=i;
- for(int i=;i<=n;i++) if(sa[i]>j) k[++p]=sa[i]-j;
- for(int i=;i<=m;i++) c[i]=;
- for(int i=;i<=n;i++) c[r[k[i]]]++;
- for(int i=;i<=m;i++) c[i]+=c[i-];
- for(int i=n;i>=;i--) sa[c[r[k[i]]]--]=k[i];
- swap(r,k);p=;r[sa[]]=++p;
- for(int i=;i<=n;i++) r[sa[i]]=cmp(k,sa[i],sa[i-],j)?p:++p;
- if(p>=n) break;m=p;
- }
- }
- struct edge{
- int v,ne;
- }e[N];
- int h[N],cnt;
- inline void ins(int u,int v){
- cnt++;
- e[cnt].v=v;e[cnt].ne=h[u];h[u]=cnt;
- }
- int fa[N],mn[N],mx[N];
- int find(int x){return x==fa[x]?x:fa[x]=find(fa[x]);}
- void unn(int x,int y){
- x=find(x);y=find(y);
- if(x!=y){
- fa[x]=y;
- mn[y]=min(mn[y],mn[x]);
- mx[y]=max(mx[y],mx[x]);
- }
- }
- void solve(){
- cnt=;memset(h,,sizeof(h));
- for(int i=;i<=n;i++){
- fa[i]=i,mn[i]=mx[i]=sa[i];
- ins(height[i],i);
- }
- for(int L=n-;L>=;L--){
- for(int i=h[L];i;i=e[i].ne) unn(e[i].v,e[i].v-);
- if(h[L]){
- int x=find(e[h[L]].v);
- if(mx[x]-mn[x]>L) {printf("%d\n",L+);return;}
- }
- }
- puts("");
- }
- int main(){
- freopen("in","r",stdin);
- while((n=read())){
- n--;
- int last=read(),x;
- for(int i=;i<=n;i++){
- x=read();
- s[i]=x-last+;
- last=x;
- }
- getSA();
- getHeight();
- solve();
- }
- }
SA+并查集
- #include <iostream>
- #include <cstdio>
- #include <cstring>
- using namespace std;
- const int N=2e4+;
- inline int read(){
- char c=getchar();int x=,f=;
- while(c<''||c>''){if(c=='-')f=-; c=getchar();}
- while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
- return x*f;
- }
- int n,m,k;
- int s[N];
- int sa[N],c[N],t1[N],t2[N];
- inline bool cmp(int *r,int a,int b,int j){
- return a+j<=n&&b+j<=n&&r[a]==r[b]&&r[a+j]==r[b+j];
- }
- int rnk[N],height[N];
- void getHeight(){
- int k=;
- for(int i=;i<=n;i++) rnk[sa[i]]=i;
- for(int i=;i<=n;i++){
- if(k) k--;
- if(rnk[i]==) continue;
- int j=sa[rnk[i]-];
- while(i+k<=n&&j+k<=n&&s[i+k]==s[j+k]) k++;
- height[rnk[i]]=k;
- }
- }
- void buildSA(){
- int *r=t1,*y=t2;
- for(int i=;i<=m;i++) c[i]=;
- for(int i=;i<=n;i++) c[r[i]=s[i]]++;
- for(int i=;i<=m;i++) c[i]+=c[i-];
- for(int i=n;i>=;i--) sa[c[r[i]]--]=i;
- for(int j=;j<=n;j<<=){
- int p=;
- for(int i=n-j+;i<=n;i++) y[++p]=i;
- for(int i=;i<=n;i++) if(sa[i]>j) y[++p]=sa[i]-j;
- for(int i=;i<=m;i++) c[i]=;
- for(int i=;i<=n;i++) c[r[y[i]]]++;
- for(int i=;i<=m;i++) c[i]+=c[i-];
- for(int i=n;i>=;i--) sa[c[r[y[i]]]--]=y[i];
- swap(r,y);p=;r[sa[]]=++p;
- for(int i=;i<=n;i++) r[sa[i]]=cmp(y,sa[i],sa[i-],j)?p:++p;
- if(p>=n) break;m=p;
- }
- getHeight();
- }
- bool check(int mid){
- int mn=sa[],mx=sa[];
- for(int i=;i<=n;i++){
- if(height[i]>=mid){
- mn=min(mn,sa[i]);
- mx=max(mx,sa[i]);
- if(mx-mn>mid) return true;
- }else mn=mx=sa[i];
- }
- return false;
- }
- void solve(){
- int l=,r=n>>,ans=;
- while(l<=r){
- int mid=(l+r)>>;
- if(check(mid)) ans=mid,l=mid+;
- else r=mid-;
- }
- if(ans+<) puts("");
- else printf("%d\n",ans+);
- }
- int main(){
- //freopen("in.txt","r",stdin);
- while((n=read())){
- n--;m=;
- int last=read(),x;
- for(int i=;i<=n;i++){
- x=read();
- s[i]=x-last+;
- last=x;
- }
- buildSA();
- solve();
- }
- }
POJ1743 Musical Theme [后缀数组]的更多相关文章
- POJ1743 Musical Theme —— 后缀数组 重复出现且不重叠的最长子串
题目链接:https://vjudge.net/problem/POJ-1743 Musical Theme Time Limit: 1000MS Memory Limit: 30000K Tot ...
- POJ1743 Musical Theme [后缀数组+分组/并查集]
Musical Theme Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 27539 Accepted: 9290 De ...
- POJ1743 Musical Theme(后缀数组 二分)
Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 33462 Accepted: 11124 Description A m ...
- POJ-1743 Musical Theme(后缀数组)
题目大意:给一个整数序列,找出最长的连续变化相同的.至少出现两次并且不相重叠一个子序列. 题目分析:二分枚举长度进行判定. 代码如下: # include<iostream> # incl ...
- poj1743 Musical Theme 后缀数组的应用(求最长不重叠重复子串)
题目链接:http://poj.org/problem?id=1743 题目理解起来比较有困难,其实就是求最长有N(1 <= N <=20000)个音符的序列来表示一首乐曲,每个音符都是1 ...
- [Poj1743] [后缀数组论文例题] Musical Theme [后缀数组不可重叠最长重复子串]
利用后缀数组,先对读入整数处理str[i]=str[i+1]-str[i]+90这样可以避免负数,计算Height数组,二分答案,如果某处H<lim则将H数组分开,最终分成若干块,判断每块中是否 ...
- POJ 1743 Musical Theme 后缀数组 最长重复不相交子串
Musical ThemeTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://poj.org/problem?id=1743 Description ...
- poj 1743 Musical Theme (后缀数组+二分法)
Musical Theme Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 16162 Accepted: 5577 De ...
- Poj 1743 Musical Theme(后缀数组+二分答案)
Musical Theme Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 28435 Accepted: 9604 Descri ...
随机推荐
- C# ini文件操作【源码下载】
介绍C#如何对ini文件进行读写操作,C#可以通过调用[kernel32.dll]文件中的 WritePrivateProfileString()和GetPrivateProfileString()函 ...
- 线性判别分析LDA原理总结
在主成分分析(PCA)原理总结中,我们对降维算法PCA做了总结.这里我们就对另外一种经典的降维方法线性判别分析(Linear Discriminant Analysis, 以下简称LDA)做一个总结. ...
- ArcGIS 10.0紧凑型切片读写方法
首先介绍一下ArcGIS10.0的缓存机制: 切片方案 切片方案包括缓存的比例级别.切片尺寸和切片原点.这些属性定义缓存边界的存在位置,在某些客户端中叠加缓存时匹配这些属性十分重要.图像格式和抗锯齿等 ...
- 为什么 NaN 不等于自身?
NaN 即Not a Number , 不是一个数字, 那么NaN到底是什么呢? 话说在JavaScript中,有6大数据类型,分别包括string,number,boolean,undefined, ...
- Ubuntu 16.10 安装byzanz截取动态效果图工具
1.了解byzanz截取动态效果图工具 byzanz能制作文件小,清晰的GIF动态效果图,不足就是,目前只能通过输入命令方式来录制. byzanz主要的参数选项有: -d, --duration=SE ...
- 浅谈web攻防
CSRF 跨站请求伪造(Cross-Site Request Forgery) -原理- 从上图可以看出,要完成一次CSRF攻击,受害者必须依次完成两个步骤: 1.登录受信任网站A,并在本地生成Coo ...
- vs2010静态链接MFC库报链接错误
由于需要将MFC程序在其它电脑上运行,所以需要将动态链接的MFC改成静态链接,本以为很简单,没想到链接的时候出现下面的链接错误: uafxcw.lib(afxmem.obj) : error LNK2 ...
- IE8/9 本地预览上传图片
本地预览的意思是,在选择图片之后先不上传到服务器,而是由一个<img>标签来预览本地的图片,非 IE8/9 浏览器可以从<input type="file"/&g ...
- 接口--interface
“interface”(接口)关键字使抽象的概念更深入了一层.我们可将其想象为一个“纯”抽象类.它允许创建者规定一个类的基本形式:方法名.自变量列表以及返回类型,但不规定方法主体.接口也包含了基本数据 ...
- Android 在Android代码中执行命令行
1.路径最好不要是自己拼写的路径/mnt/shell/emulated/0/wifidog.conf 最好是通过方法获取的路径,不然可能导致命令无效 (挂载点的原因) public static f ...