51nod 1732 51nod婚姻介绍所 后缀数组:最长公共前缀

2023-11-09 23:48

本文主要是介绍51nod 1732 51nod婚姻介绍所 后缀数组:最长公共前缀,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1732 51nod婚姻介绍所

  1. 1.0 秒
  2.  
  3. 131,072.0 KB
  4.  
  5. 40 分
  6.  
  7. 4级题

51nod除了在做OJ之外,还开展了很多副业。婚姻介绍所就是其中之一。

对于一个客户,我们可以使用一个字符串来描述该客户的特质。

假设现在我们有两个客户A和B。

A的特质字符串为:abcdefg

B的特质字符串为:abcxyz

则A和B的匹配度f(A, B)为A和B的最长公共前缀的长度,即len('abc') = 3

由于最近51nod经费紧张,所以夹克大老爷设计了一种压缩算法以节约内存。

所有用户的特质字符串都被存储在了一个长为n的字符串S中。(n <= 1000)用户的特质使用一个整数p表示,表示该用户的特质字符串为S[p...n - 1]。

现给定字符串S,与q次查询<ai, bi>(ai, bi分别为合法的用户特质整数)。请输出q次查询分别对应的客户匹配度。

 

 收起

输入

现给定字符串长度n,与字符串S。接下来是整数q,代表接下来有q次查询。
下面q行有两个整数ai, bi。代表查询特质为ai与bi的用户的匹配度。1 <= n <= 1000
1 <= q <= 10^6输入数据全部合法。

输出

每一行输出一个用户匹配度整数。

输入样例

12
loveornolove
5
3 7
0 0
9 1
3 1
9 5

输出样例

0
12
3
0
0

 

分析:

后缀数组的模板题,注意快读

#include <bits/stdc++.h>
using namespace std;
const int maxn=200000+1000;
int len1,len2;
void in(int &x)
{x=0;char c=getchar();while(c<'0'||c>'9'){c=getchar();}x=c-'0';while(c=getchar()){if(c<'0'||c>'9') break;x=x*10+c-'0';}
}void out(int x)
{if(x>=10)out(x/10);putchar(x%10+'0');
}
struct SuffixArray
{char s[maxn];///_rank[i] 第i个后缀的排名; SA[i] 排名为i的后缀位置; Height[i] 排名为i的后缀与排名为(i-1)的后缀的LCPint sa[maxn],_rank[maxn],height[maxn];///c[i] 基数排序辅助数组int t1[maxn],t2[maxn],c[maxn],n;int dmin[maxn][21];void init(){memset(height,0,sizeof(height));memset(_rank,0,sizeof(_rank));memset(sa,0,sizeof(sa));memset(c,0,sizeof(c));memset(t1,0,sizeof(t1));memset(t2,0,sizeof(t2));memset(dmin,0,sizeof(dmin));}void build_sa(int m)  ///m大于s[]数组出现的任意字符的int值{/// x[i]是第i个元素的第一关键字  y[i]表示第二关键字排名为i的数,第一关键字的位置int i,p,*x=t1,*y=t2;x[n]=y[n]=-1;for(i=0; i<m; i++)c[i]=0;for(i=0; i<n; i++)c[x[i]=s[i]]++;for(i=1; i<m; i++)c[i]+=c[i-1];for(i=n-1; i>=0; i--)sa[--c[x[i]]]=i;for(int k=1; k<=n; k<<=1){p=0;for(i=n-k; i<n; i++)y[p++]=i;for(i=0; i<n; i++)if(sa[i]>=k)y[p++]=sa[i]-k;for(i=0; i<m; i++)c[i]=0;for(i=0; i<n; i++)c[x[i]]++;for(i=1; i<m; i++)c[i]+=c[i-1];for(i=n-1; i>=0; i--)sa[--c[x[y[i]]]]=y[i];swap(x,y);p=1;x[sa[0]]=0;for(i=1; i<n; i++){if(y[sa[i]]==y[sa[i-1]]&&y[sa[i]+k]==y[sa[i-1]+k])x[sa[i]]=p-1;elsex[sa[i]]=p++;}if(p>=n)break;m=p;}}void build_height()//单个字符也行{int i,j,k=0,r;for(i=0; i<n; i++)_rank[sa[i]]=i;height[0]=0;for(i=0; i<n; i++){if(k)k--;r=_rank[i];if(r==0)continue;j=sa[r-1];while(s[i+k]==s[j+k])k++;height[_rank[i]]=k;}}int LongestMessage() //最长公共子串{int ans=0;for(int i=2; i<n; i++){int a1=sa[i-1],a2=sa[i];if(a1>a2)swap(a1,a2);if(a1>=0&&a1<=len1-1&&a2>=len1+1&&a2<=len1+len2)ans = max(ans,height[i]);}return ans;}void initMin(){for(int i=0; i<n; i++)dmin[i][0]=height[i];for(int j=1; (1<<j)<=n; j++)for(int i=0; i+(1<<j)-1<n; i++)dmin[i][j]=min(dmin[i][j-1],dmin[i+(1<<(j-1))][j-1]);}int RMQ(int L,int R)//取得范围最小值{int k=0;while((1<<(k+1))<=R-L+1)k++;return min(dmin[L][k], dmin[R-(1<<k)+1][k]);}int LCP(int i,int j)//求后缀i和j的LCP最长公共前缀{if(i==j) return n-i;int L=_rank[i],R=_rank[j];//求后缀i与后缀j的LCP//if(i==j) return n-sa[i];//int L=i,R=j;//直接求排名i与j后缀的LCPif(L>R)swap(L,R);L++;//注意这里return RMQ(L,R);}int num()//子串的个数{int ans=0;for(int i=1; i<n; i++)ans += n-1-sa[i]-height[i];return ans;}//确定子串[be,be+len-1]的在后缀排名区间[L,R]int get_LR(int be,int len,int &L,int &R){int pos=_rank[be];int l=0,r=pos;while(l<r){int mid=(l+r)>>1;if(RMQ(mid+1,pos)>=len)r=mid;elsel=mid+1;}L=l;l=pos;r=n-1;while(l<r){int mid=(l+r+1)>>1;if(RMQ(pos+1,mid)>=len)l=mid;elser=mid-1;}R=l;}//恰好出现w次子串的个数int num_w(int w){int ans=0;for(int i=0; i+w-1<n; i++)ans+=max(0,LCP(i,i+w-1)-max(height[i],height[i+w]));return ans;}void out(){for(int i=0; i<n; i++){cout<<sa[i]<<" ";}cout<<endl;for(int i=0; i<n; i++){cout<<_rank[i]<<" ";}cout<<endl;for(int i=0; i<n; i++){cout<<height[i]<<" ";}cout<<endl;}
} sa;
int main()
{sa.init();scanf("%d",&len1);scanf("%s",sa.s);sa.n=len1;sa.build_sa(256);sa.build_height();sa.initMin();int q;in(q);int a,b;while(q--){in(a);in(b);if(a==b)out(len1-a);elseout(sa.LCP(a,b));puts("");}return 0;
}

 

 

这篇关于51nod 1732 51nod婚姻介绍所 后缀数组:最长公共前缀的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/379172

相关文章

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

Python在固定文件夹批量创建固定后缀的文件(方法详解)

《Python在固定文件夹批量创建固定后缀的文件(方法详解)》文章讲述了如何使用Python批量创建后缀为.md的文件夹,生成100个,代码中需要修改的路径、前缀和后缀名,并提供了注意事项和代码示例,... 目录1. python需求的任务2. Python代码的实现3. 代码修改的位置4. 运行结果5.

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

poj3261(可重复k次的最长子串)

题意:可重复k次的最长子串 解题思路:求所有区间[x,x+k-1]中的最小值的最大值。求sa时间复杂度Nlog(N),求最值时间复杂度N*N,但实际复杂度很低。题目数据也比较水,不然估计过不了。 代码入下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring

poj1330(LCA最近公共祖先)

题意:求最近公共祖先 思路:之前学习了树链剖分,然后我就用树链剖分的一小部分知识就可以解这个题目了,记录每个结点的fa和depth。然后查找时,每次将depth大的结点往上走直到x = y。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring>

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

hihocoder1050 : 树中的最长路

时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 上回说到,小Ho得到了一棵二叉树玩具,这个玩具是由小球和木棍连接起来的,而在拆拼它的过程中,小Ho发现他不仅仅可以拼凑成一棵二叉树!还可以拼凑成一棵多叉树——好吧,其实就是更为平常的树而已。 但是不管怎么说,小Ho喜爱的玩具又升级换代了,于是他更加爱不释手(其实说起来小球和木棍有什么好玩的是吧= =)。小Ho手