BZOJ 3172([Tjoi2013]单词-后缀数组第一题+RMQ)
3172: [Tjoi2013]单词
Time Limit: 10 Sec Memory Limit: 512 MB
Submit: 268
Solved: 145
[
Submit][
Status]
Description
某人读论文,一篇论文是由许多单词组成。但他发现一个单词会在论文中出现很多次,现在想知道每个单词分别在论文
中出现多少次。
Input
第一个一个整数N,表示有多少个单词,接下来N行每行一个单词。每个单词由小写字母组成,N<=200,单词长度不超过10^6
Output
输出N个整数,第i行的数字表示第i个单词在文章中出现了多少次。
Sample Input
a
aa
aaa
Sample Output
3
1
上一次写RMQ是什么时候?(喂,离题了)
好吧……
第一题后缀数组
不想写下去了……(快哭了TNT)
这题在BZOJ上内存很容易开过(5人组-》TLE/CE/MLE/RE/AC)
大家要是这题RE把数组开小点。别忘了[RMQ*20]数组+数组之和 //省空间
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<functional>
#include<cmath>
#include<cctype>
using namespace std;
#define For(i,n) for(int i=1;i<=n;i++)
#define Rep(i,n) for(int i=0;i<n;i++)
#define Fork(i,k,n) for(int i=k;i<=n;i++)
#define ForD(i,n) for(int i=n;i;i--)
#define Forp(x) for(int p=pre[x];p;p=next[p])
#define RepD(i,n) for(int i=n;i>=0;i--)
#define MEM(a) memset(a,0,sizeof(a))
#define MEMI(a) memset(a,127,sizeof(a))
#define MEMi(a) memset(a,128,sizeof(a))
#define INF (2139062143)
#define F (1000000009)
#define MAXN (300+10)
#define MAXL (1000200+10)
#define eps (1e-9)
typedef long long ll;
char s[MAXL];
int n,pre[MAXN],tai[MAXN];
int w[MAXL],sa[MAXL],wa[MAXL*2]={0},wb[MAXL*2]={0};
// x-->上一行 y->下一行sa右值 wv-->y的左值 sa-->上次排名(求)
bool cmp(int *a,int x,int y,int l){return (a[x]==a[y]&&a[x+l]==a[y+l]);}
void suffix_array(int n,int m)
{
int *x=wa,*y=wb;
For(i,m) w[i]=0;
For(i,n) w[x[i]=s[i]]++;
Fork(i,2,m) w[i]+=w[i-1];
ForD(i,n) sa[w[x[i]]--]=i;
for(int j=1,p=0;p<n;j*=2,m=p)
{ p=0;
Fork(i,n-j+1,n) y[++p]=i;
For(i,n) if (j<sa[i]) y[++p]=sa[i]-j; For(i,m) w[i]=0;
For(i,n) w[x[i]]++;
For(i,m) w[i]+=w[i-1]; ForD(i,n) sa[w[x[ y[i] ]]--]=y[i];
//y is release p=y[sa[1]]=1;
Fork(i,2,n)
y[sa[i]]=(p+=(!cmp(x,sa[i-1],sa[i],j))); int *t=x;x=y;y=t;
}
}
int height[MAXL],rank[MAXL]; //height[i] 表示 sa[i]与sa[i-1]的最长公共前缀
void make_height(char *s,int n)
{
For(i,n) rank[sa[i]]=i;
For(i,n)
{
if (rank[i]==1) continue; //求height[rank[i]]
int j=max(0,height[rank[i-1]]-1),k=sa[rank[i]-1];
while (s[i+j]==s[k+j]) j++;
height[rank[i]]=j;
}
}
int bin[MAXN]={0},f[MAXL][24]={0};
int lcp(int l,int r)
{
int len=r-l+1,j=(int)log2(len);
return min(f[l][j],f[r-bin[j]+1][j]);
}
int main()
{
// freopen("bzoj3172.in","r",stdin);
scanf("%d",&n);pre[1]=1;
For(i,n)
{
scanf("%s",s+pre[i]);
tai[i]=strlen(s+pre[i]);
s[pre[i]+tai[i]]='#';
pre[i+1]=pre[i]+tai[i]+1;
}
s[pre[n+1]]=0; suffix_array(pre[n+1]-1,200); make_height(s,pre[n+1]-1); int logn=int(log2((double)pre[n+1]-1)+1);
bin[0]=1;
For(i,logn) bin[i]=bin[i-1]<<1;
For(i,pre[n+1]-1) f[i][0]=height[i]; For(j,logn)
For(i,pre[n+1]-1-(1<<j)+1)
f[i][j]=min(f[i][j-1],f[i+(1<<(j-1))][j-1]); For(i,n)
{
int tot=1;
int j=rank[pre[i]]+1;
{
int l=2,r=j-1,ans=0;
while (l<=r)
{
int m=l+r>>1;
if (lcp(m,j-1)>=tai[i]) ans=m,r=m-1;
else l=m+1;
}
if (ans) ans=j-1-ans+1;
tot+=ans;
}
{
int l=j,r=pre[n+1]-1,ans=0;
while (l<=r)
{
int m=l+r>>1;
if (lcp(j,m)>=tai[i]) ans=m,l=m+1;
else r=m-1;
}
if (ans) ans=ans-j+1;
tot+=ans;
}
cout<<tot<<endl;
}
// while (1);
return 0;
}
BZOJ 3172([Tjoi2013]单词-后缀数组第一题+RMQ)的更多相关文章
- BZOJ 3172 Tjoi2013 单词 后缀数组
题目大意:给定一个n个单词的文章,求每一个单词在文章中的出现次数 文章长度<=10^6(不是单词长度<=10^6,不然读入直接超时) 首先将全部单词用空格连接成一个字符串.记录每一个单词的 ...
- BZOJ 3172: [Tjoi2013]单词 [AC自动机 Fail树]
3172: [Tjoi2013]单词 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 3198 Solved: 1532[Submit][Status ...
- bzoj 3172: [Tjoi2013]单词 AC自动机
3172: [Tjoi2013]单词 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...
- ●BZOJ 3172 [Tjoi2013]单词
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=3172 题解: 把单词逐个接起来,中间用互不相同的字符连接,并记录下每个单词的首字母在串中的位 ...
- [BZOJ 3172] [Tjoi2013] 单词 【AC自动机】
题目链接:BZOJ - 3172 题目分析: 题目要求求出每个单词出现的次数,如果把每个单词都在AC自动机里直接跑一遍,复杂度会很高. 这里使用AC自动机的“副产品”——Fail树,Fail树的一个性 ...
- BZOJ 3172 [Tjoi2013]单词 AC自己主动机(fail树)
题意:链接 方法:AC自己主动机与fail树性质 解析:复习AC自己主动机的第一道题?(真正的第一题明明是又一次写了遍hdu2222! ) 这题说实话第一眼看上去就是个sb题,仅仅要建出来自己主动机. ...
- 【刷题】BZOJ 3172 [Tjoi2013]单词
Description 某人读论文,一篇论文是由许多单词组成.但他发现一个单词会在论文中出现很多次,现在想知道每个单词分别在论文中出现多少次. Input 第一个一个整数N,表示有多少个单词,接下来N ...
- bzoj 3172 [Tjoi2013]单词(fail树,DP)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3172 [题意] 题目的意思是这样的,给若干个单词,求每个单词在这一堆单词中的出现次数. ...
- BZOJ 3172 [Tjoi2013]单词 AC自动机Fail树
题目链接:[http://www.lydsy.com/JudgeOnline/problem.php?id=3172] 题意:给出一个文章的所有单词,然后找出每个单词在文章中出现的次数,单词用标点符号 ...
随机推荐
- Swift - 使用UIDatePicker实现倒计时功能
如果使用UIDatePicker时将模式设置为CountDownTimer,即可让该控件作为倒计时器来使用.效果图如下: 下面是代码示例: 1 2 3 4 5 6 7 8 9 10 11 12 ...
- iOS - NSLog的使用方法
NSLog的定义 NSLog定义在NSObjCRuntime.h中,如下所示: void NSLog(NSString *format, …); 基本上,NSLog很像printf,同样会在conso ...
- Win7和VS2013上使用Intel的TBB
源地址:http://www.th7.cn/system/win/201505/103966.shtml http://wenku.baidu.com/link?url=zH7vwmWltWF5R-9 ...
- IOT表优缺点
<pre name="code" class="html">IOT表是将所有东西都塞到叶块中,表就是索引,可以避免回表 首先,对于IOT而言,只有索 ...
- URAL 1963 Kite 四边形求对称轴数
题目链接: http://acm.timus.ru/problem.aspx?space=1&num=1963 题意,顺时针或逆时针给定4个坐标,问对称轴有几条,输出(对称轴数*2) 对于一条 ...
- 用wireshark解析应用层存储包
工作中常常须要统计server上的rtp包接收.发送性能.不想自己再做一套统计软件,打算用现有的wireshark来做分析统计. 先把rtp头存成pcap格式文件,pcap文件格式及如何存储能够參照这 ...
- 冒泡排序 JAVA版
冒泡排序 算法思想是每次从数组末端开始比较相邻俩元素,把第i小的冒泡到数组的第i个位置.i从0一直到N-1从而完成排序.当然也可以从数组开始端开始比较相邻两元素,把第i大的冒泡到第N-i个位置.I从0 ...
- UNIX 网络编程之线程
概述: 实现并发服务器一般都是父进程accept一个连接,然后fork一个子进程,该子进程处理与该连接对端的客户之间的通信.但是fork是昂贵,耗资源和时间.而线程是轻量级线程,它的创建比进程的创建块 ...
- hdu1896之优先队列应用
Stones Time Limit: 5000/3000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others) Total Sub ...
- currentstyle和getComputedStyle兼容问题
currentStyle:获取计算后的样式,也叫当前样式.终于样式. 长处:能够获取元素的终于样式,包含浏览器的默认值,而不像style仅仅能获取行间样式,所以更经常使用到. 注意:不能获取复合样式如 ...