翻译下题解…
可以发现一次变化是
f(SS)=STST
,其中
T
是S的一个前缀,
那么我们就只需要考虑变化
g(S)=ST
,因为不断变化
10100
次的话,后面的部分就可以省去
设
T
的长度为
可以证明如果
d|n
,那么
g(ST)=STT
否则,
g(ST)=STS
那么分类讨论一下,瞎搞一搞就可以了
#include <cstdio>
#include <iostream>
#include <algorithm>
#include <vector>
#include <string>
#include <cstring>
using namespace std;
const int N=200010;
typedef long long ll;
int n,prd;
int fail[N];
char a[N];
ll l,r,pre[N][30];
inline ll fib(ll p,int c){
if(p<=n) return pre[p][c];
if(p<=n*2) return pre[n][c]+pre[p-n][c];
ll ret1=pre[n][c]+pre[prd][c],ret2=pre[n][c],a=n+prd,b=n;
while(p>a+b){
ll c=ret1; ret1+=ret2; ret2=c;
c=a; a+=b; b=c;
}
return ret1+fib(p-a,c);
}
inline ll calc(ll p,int c){
if(p<=n) return pre[p][c];
if(n%prd==0){
ll ret=1LL*(p-n)/prd*pre[prd][c]+pre[n][c];
p-=n; p%=prd;
return ret+pre[p][c];
}
else return fib(p,c);
}
int main(){
scanf("%s",a+1); n=strlen(a+1)/2;
for(int i=1;i<=n;i++){
for(int j=0;j<26;j++) pre[i][j]=pre[i-1][j];
pre[i][a[i]-'a']++;
}
scanf("%lld%lld",&l,&r);
int k=0;
for(int i=2;i<=n;i++){
while(k&&a[k+1]!=a[i]) k=fail[k];
if(a[k+1]==a[i]) k++;
fail[i]=k;
}
prd=n-fail[n];
for(int i=0;i<26;i++)
printf("%lld ",calc(r,i)-calc(l-1,i));
return 0;
}