题面

切此题建议配的 BGM:おしおき feat.アーケードラビット(七海千秋处刑专用)

七海千秋是希望之峰学院 77 级 1 班的班长,其能力为 “超高校级的游戏玩家”,擅长除了恋爱游戏以外的全类型游戏,尤其喜欢 $gyara-\omega$一类的弹幕射击游戏。

最近七海又在攻略一款弹幕射击游戏,这一关对面有 $n$个敌人排成一排,每个敌人都有一个防御力 $a_i$。七海的技能是可以发射一片攻击力为 $X$的弹幕,攻击一个区间内的敌人,对于被攻击的敌人 $i$,若 $a_i<X$,该敌人死亡。如果七海发射一次弹幕能够杀死恰好 $k$个敌人,就能获得额外得分。七海想知道,有多少种选择区间的方案,可以达成这个目标。

身为 “超高校级的游戏玩家”,七海当然知道怎么算啦,但是她还要专心打游戏没时间计算,于是她写了一个用于计算答案的伪代码,交由 “超高校级的程序员”——你来实现:

for k<-0 to n do
    ans <- 该 k 情况下的答案
    print ans

题目分析

本题是给大家在做了很多神级思考题后放松一下的 FFT 板子题。
七海还是很可爱的,虽然我觉得 XZY 这个喜新厌旧的男人过几天就会把登录界面的七海妹子换掉。
显然将所有小于 $x$的数转化为 1,其他的为 0,这就是个 01 序列,然后求区间和是 $k$的区间个数。
那么我们首先要取一个左端点,再取一个右端点。
$k$要取遍,并且是 $10^5$级的数据,考虑 FFT。
考虑左端点取每一个点时,在这个点左边的 $1$的个数,若个数为 $t$,则 $x^t$的系数加 1,这样构造一个多项式 $A$。
右端点取每一个点时,在这个点右边的 $1$的个数,构造一个类似的多项式 $B$。
那么 $A$和 $B$做卷积,第 $t$项的系数就是区间外 $1$的个数为 $t$的区间个数。
桥豆麻袋,有可能出现右端点取得比左端点小的情况。
发现这种情况下,算出来在区间外 $1$的个数一定大于等于整个序列 $1$的个数,所以受影响的只有 $k=0$的情况,所以单独用组合的方法计算一下 $k=0$的情况即可。

代码

#include<bits/stdc++.h>
using namespace std;
#define RI register int
int read() {
    int q=0,w=1;char ch=' ';
    while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
    if(ch=='-') w=-1,ch=getchar();
    while(ch>='0'&&ch<='9') q=q*10+ch-'0',ch=getchar();
    return q*w;
}
typedef long long LL;
typedef double db;
const db pi=3.1415926535897384626;
const int N=524300;
struct com{db r,i;}a[N],b[N];
int n,X,tot,len,kn;
int sum[N],rev[N];LL ans[N];
com operator* (com a,com b) {return (com){a.r*b.r-a.i*b.i,a.r*b.i+a.i*b.r};}
com operator- (com a,com b) {return (com){a.r-b.r,a.i-b.i};}
com operator+ (com a,com b) {return (com){a.r+b.r,a.i+b.i};}
com operator/ (com a,db b) {return (com){a.r/b,a.i/b};}
void NTT(com *a,int n,int x) {
    for(RI i=0;i<n;++i) if(rev[i]>i) swap(a[i],a[rev[i]]);
    for(RI i=1;i<n;i<<=1) {
        com wn=(com){cos(pi/i),sin(pi/i)*x};
        for(RI j=0;j<n;j+=(i<<1)) {
            com t1,t2,w=(com){1,0};
            for(RI k=0;k<i;++k,w=w*wn)
                t1=a[j+k],t2=w*a[j+i+k],a[j+k]=t1+t2,a[j+i+k]=t1-t2;
        }
    }
    if(x==-1) for(RI i=0;i<n;++i) a[i]=a[i]/(db)n;
}
int main()
{
    n=read(),X=read();
    for(RI i=1;i<=n;++i) sum[i]=sum[i-1]+(read()<X);
    for(RI i=1;i<=n;++i) ++a[sum[i-1]].r;
    for(RI i=1;i<=n;++i) ++b[sum[n]-sum[i]].r;
    kn=1;while(kn<=sum[n]*2) kn<<=1,++len;
    for(RI i=1;i<kn;++i) rev[i]=(rev[i>>1]>>1)|((i&1)<<(len-1));
    NTT(a,kn,1),NTT(b,kn,1);
    for(RI i=0;i<kn;++i) a[i]=a[i]*b[i];
    NTT(a,kn,-1);
    int js=0;
    for(RI i=1;i<=n;++i)
        if(sum[i]==sum[i-1]) ++js;
        else ans[0]+=1LL*js*(js-1)/2+js,js=0;
    ans[0]+=1LL*js*(js-1)/2+js;
    for(RI i=0;i<sum[n];++i) ans[sum[n]-i]=(LL)(a[i].r+0.2);
    for(RI i=0;i<=n;++i) printf("%lld ",ans[i]);
    puts("");
    return 0;
}
分类: 文章

litble

苟...苟活者在淡红的血色中,会依稀看见微茫的希望

0 条评论

发表回复

Avatar placeholder

您的邮箱地址不会被公开。 必填项已用 * 标注