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] 题意:给出一个文章的所有单词,然后找出每个单词在文章中出现的次数,单词用标点符号 ...
随机推荐
- Mac OS升级到Yosemite后一些问题
苹果"优山美地"採用移动设备平面风格,看起来还是相当清爽. 只是升级完还是有一些程序兼容性问题的. 1. 开发Android的程序猿们,Java se 6 须要升级到2014_00 ...
- Html 内嵌 选择器属性 Dom操作 JavaScript 事件
HTML标签: 一.通用标签(一般标签) 1.格式控制标签 <font color="#6699aa" face="楷体" size="24&q ...
- js 常用正则表达式分析详解
1.整数或者小数:/^((0{1}|[1-9]{1}[0-9]+)\.{1}[0-9]+|[1-9]{1}[0-9]*|0)$/ 分析:分类讨论,如果是小数,则有两种形式 0.111对应的是 0{ ...
- iOS 开发百问(6)
61.警告"addexplicit braces to avoid dangling else" 所谓"危急的else"是相似这种代码: if(a== 10) ...
- php 用递归实现的无限级别分类
<?php header("Content-type:text/html; charset=utf-8"); /** * * @category contry_cate ...
- 2015年十大热门Android开源新项目
2015年十大热门Android开源新项目 2015 即将结束,又到了大家喜闻乐见的年终盘点时刻啦,今天给大家盘点一下 2015 年 Android 开发领域新出现的 10 大热门开源项目.数据来自于 ...
- JSON支持什么对象/类型?
原文:JSON支持什么对象/类型? 当一个面试官问你: JSON都支持什么对象/类型?你怎么回答? 也许他的本意是下面这个答案: JSON格式支持的数据类型有以下: 类型 描述 Number 在Jav ...
- Android-1-电话拨号程序
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdTAxMjY1MTM4OQ==/font/5a6L5L2T/fontsize/400/fill/I0JBQk ...
- ftk学习记(waitbox篇)
[声明:版权全部.欢迎转载,请勿用于商业用途. 联系信箱:feixiaoxing @163.com] 前面说到了脚本.那么就看看ftk中demo与script搭配的效果是什么样的? 上面的效果图就相 ...
- poj2777--Count Color(线段树,二进制转化)
Count Color Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 34950 Accepted: 10542 Des ...