比赛的时候我是用后缀数组的,但是T了。
赛后看了解题报告说,后缀数组貌似是卡你常数的时间,我算了下复杂度O(T * Q * n)。这是10 ^ 8,但是考虑到每次询问的时候都要重新构造字符,所以那个n可能是(3 - 4 ) * n,卡的可能就是这个常数。然后就过不了了。
我先上一发我的后缀数组的代码,T的好惨。
因为当时不会后缀自动机。
#include
#include
#include
#include
#include
using namespace std;
const int N=2005;
/****后缀数组模版****/
#define F(x)((x)/3+((x)%3==1 0:tb)) //F(x)求出原字符串的suffix(x)在新的字符串中的起始位置
#define G(x)((x)=0; i--)
b[--WS[wv[i]]]=a[i];
return;
}
//注意点:为了方便下面的递归处理,r数组和sa数组的大小都要是3*n
void dc3(int *r,int *sa,int n,int m) { //rn数组保存的是递归处理的新字符串,san数组是新字符串的sa
int i , j , *rn = r+n , *san = sa+n , ta = 0 ,tb = (n+1)/3 , tbc = 0 , p;
r[n] = r[n+1] = 0;
for(i=0; i '9') ;
ret = c - '0';
while((c=getchar()) >= '0' && c <= '9')
ret = ret * 10 + ( c - '0' );
}
inline void OT(int a) {
if(a >= 10)OT(a / 10) ;
putchar(a % 10 + '0') ;
}
struct QU {
int s ,e ,id ;
} QQ[10005] ;
bool cmp(QU a ,QU b) {
if(a.s == b.s)return a.e < b.e ;
return a.s < b.s ;
}
int ans[11111] ;
int main() {
int i,n,t;
scanf("%d",&t);
while(t--) {
scanf("%s",str);
memset(ans , 0 , sizeof(ans));
int xd ;
cin >> xd ;
for (int i = 0 ; i < xd ; i ++ ) {
RD(QQ[i].s) ;
RD(QQ[i].e) ;
QQ[i].s -- ;
QQ[i].e -- ;
QQ[i].id = i ;
}
sort(QQ , QQ + xd , cmp) ;
int st = QQ[0].s ;
int ed = QQ[0].e ;
int num = 0 ;
for (int i = st ; i <= ed ; i ++ )r[num ++] = (int)str[i] ;
r[num] = 0 ;
dc3(r , sa ,num + 1 , 200) ;
calheight(r , sa, num ) ;
ans[QQ[0].id] = solve(num) ;
for (int i = 1 ; i < xd ; i ++ ) {
if(QQ[i].s != QQ[i - 1].s) {
num = 0 ;
for (int j = QQ[i].s ; j <= QQ[i].e ; j ++ )r[num ++] = (int)str[j] ;
r[num] = 0 ;
dc3(r , sa , num + 1 ,200 ) ;
calheight(r ,sa ,num) ;
ans[QQ[i].id] = solve(num) ;
ed = QQ[i].e ;
} else {
if(QQ[i].e == QQ[i - 1].e) {
ans[QQ[i].id] = ans[QQ[i - 1].id] ;
} else {
for (int j = ed + 1 ; j <= QQ[i].e ; j ++)r[num ++] = (int)str[j] ;
r[num] = 0 ;
dc3(r ,sa ,num + 1 , 200) ;
calheight(r ,sa ,num) ;
ans[QQ[i].id] = solve(num) ;
ed = QQ[i].e ;
}
}
}
for (int i = 0 ; i < xd ; i ++ ) {
OT(ans[i]) ;
puts("") ;
}
}
return 0;
}
#include
#include
#include
#include
#include
using namespace std;
const int N=2005;
/****后缀数组模版****/
#define F(x)((x)/3+((x)%3==1 0:tb)) //F(x)求出原字符串的suffix(x)在新的字符串中的起始位置
#define G(x)((x)=0; i--)
b[--WS[wv[i]]]=a[i];
return;
}
//注意点:为了方便下面的递归处理,r数组和sa数组的大小都要是3*n
void dc3(int *r,int *sa,int n,int m) { //rn数组保存的是递归处理的新字符串,san数组是新字符串的sa
int i , j , *rn = r+n , *san = sa+n , ta = 0 ,tb = (n+1)/3 , tbc = 0 , p;
r[n] = r[n+1] = 0;
for(i=0; i '9') ;
ret = c - '0';
while((c=getchar()) >= '0' && c <= '9')
ret = ret * 10 + ( c - '0' );
}
inline void OT(int a) {
if(a >= 10)OT(a / 10) ;
putchar(a % 10 + '0') ;
}
struct QU {
int s ,e ,id ;
} QQ[10005] ;
bool cmp(QU a ,QU b) {
if(a.s == b.s)return a.e < b.e ;
return a.s < b.s ;
}
int ans[11111] ;
int main() {
int i,n,t;
scanf("%d",&t);
while(t--) {
scanf("%s",str);
memset(ans , 0 , sizeof(ans));
int xd ;
cin >> xd ;
for (int i = 0 ; i < xd ; i ++ ) {
RD(QQ[i].s) ;
RD(QQ[i].e) ;
QQ[i].s -- ;
QQ[i].e -- ;
QQ[i].id = i ;
}
sort(QQ , QQ + xd , cmp) ;
int st = QQ[0].s ;
int ed = QQ[0].e ;
int num = 0 ;
for (int i = st ; i <= ed ; i ++ )r[num ++] = (int)str[i] ;
r[num] = 0 ;
dc3(r , sa ,num + 1 , 200) ;
calheight(r , sa, num ) ;
ans[QQ[0].id] = solve(num) ;
for (int i = 1 ; i < xd ; i ++ ) {
if(QQ[i].s != QQ[i - 1].s) {
num = 0 ;
for (int j = QQ[i].s ; j <= QQ[i].e ; j ++ )r[num ++] = (int)str[j] ;
r[num] = 0 ;
dc3(r , sa , num + 1 ,200 ) ;
calheight(r ,sa ,num) ;
ans[QQ[i].id] = solve(num) ;
ed = QQ[i].e ;
} else {
if(QQ[i].e == QQ[i - 1].e) {
ans[QQ[i].id] = ans[QQ[i - 1].id] ;
} else {
for (int j = ed + 1 ; j <= QQ[i].e ; j ++)r[num ++] = (int)str[j] ;
r[num] = 0 ;
dc3(r ,sa ,num + 1 , 200) ;
calheight(r ,sa ,num) ;
ans[QQ[i].id] = solve(num) ;
ed = QQ[i].e ;
}
}
}
for (int i = 0 ; i < xd ; i ++ ) {
OT(ans[i]) ;
puts("") ;
}
}
return 0;
}
昨天晚上和今天一直在看后缀自动机,这里推荐一个我看的懂的博客。
http://blog.sina.com.cn/s/blog_70811e1a01014dkz.html
算是学会了使用模版。当然也仅限于模版题。。好水。。继续学习。。
[cpp] view plaincopyprint
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include