zoukankan      html  css  js  c++  java
  • CQOI2018 交错序列

    这道题考察知识点的综合性。借助多方力量的帮助,我差不多把这道题弄明白了。

    题目描述

    我们称一个仅由0、1构成的序列为”交错序列“,当且仅当序列中没有相邻的1(可以有相邻的0)。例如,000,001,101,都是交错序列,而110 则不是。

    对于一个长度为n 的交错序列,统计其中0 和1出现的次数,分别记为x 和y。给定参数a、b,定义一个交错序列的特征值为(x^ay^b)。注意这里规定任何整数的0 次幂都等于1(包括(0^0=1))。

    显然长度为n 的交错序列可能有多个。我们想要知道,所有长度为n 的交错序列的特征值的和,除以m 的余数。(m 是一个给定的质数)

    对于30%的数据,1≤n≤15

    对于100%的数据,1≤n≤10000000 0≤a,b≤45 m<100000000

    1.1 思路

    设一个交错序列(lambda)的特征值为(s(lambda)),则我们要求的答案(ans=sum_{|lambda|=n}s(lambda))
    (=sum_{|lambda|=n}x^ay^b=sum_{|lambda|=n}(n-y)^ay^b)
    (=sum_{|lambda|=n}y^bsum_{k=0}^{a}dbinom{a}{k}n^ky^{a-k}(-1)^{a-k})
    (=sum_{|lambda|=n}sum_{k=0}^{a}dbinom{a}{k}n^ky^{a-k+b}(-1)^{a-k})
    我们记(f(k)=dbinom{a}{k}n^k(-1)^{a-k}),则原式可以写成(ans=sum_{|lambda|=n}sum_{k=0}^{a}f(k)y^{a-k+b})

    注意到,把这个式子展开来看:(ans=sum_{k=0}^{a}f(k)y_{lambda}^{a-k+b}+sum_{k=0}^{a}f(k)y_{lambda'}^{a-k+b}+cdots+sum_{k=0}^{a}f(k)y_{lambda^{(cdots)}}^{a-k+b})

    (=sum_{k=0}^{a}(f(k)sum_{|lambda|=n}y^{a+b-k}))

    注意到对于每一个序列,(f(k))是一个常数,可以直接把(sum)提到前面去。我们只要维护一下(sum_{|lambda|=n} y_{lambda}^{a+b-k})的值,就可以快速求出答案了。

    1.2 贡献的转移

    题目中要统计的序列是有形式上的限制的。我们可以考虑使用dp。

    我们设(f[l][p][0])表示一个以(0)结尾的,长度为(l)的序列,各序列中含有的“1”的个数的(p)次方和。也就是说,设交错序列(lambda)中含有“1”的个数为(y_{lambda}),则(f[l][p][0]=sum_{lambda_l=0}y_{lambda}^p)(f[l][p][1])同理。

    结尾为0和1的序列都可以直接在后面接0,对答案不产生贡献。由0,1结尾转移到0结尾,得到:(f[l][p][0]=f[l-1][p][0]+f[l-1][p][1])

    只有结尾为0的序列后面可以接1,对答案产生贡献。考虑贡献的转移:
    (sum{y^p} ightarrow sum{(y+1)^p}),右边展开得到:(sum{(y+1)^p}=sumsum_{k=0}^{p}dbinom{p}{k}y^k=sum_{k=0}^p{dbinom{p}{k}sum{y^k}}=sum_{k=0}^pdbinom{p}{k}f[l-1][k][0])

    这样我们就得到了全部的状态转移方程。

    1.3 转移的优化

    转移方程基本都是线性递推的,我们可以考虑使用矩阵来加速。

    第一步,由于状态(f[l])和状态(f[l-1])是独立的,我们可以考虑删去这一维。记此时的状态为(f[p][0/1])

    第二步,直接将状态(f[p][0])和状态(f[p][1])合并。由于我们最多会用到(sum y^{a+b}),我们令(S=a+b+1),设(0 leq p < S),我们可以直接令(sp[p]=f[p][0])(sp[p+S]=f[p][1])。这样整个状态用(sp[])来表示,被压缩到了一维。

    接着考虑构造转移矩阵(T[i][j])

    承袭上面的定义,转移到0结尾的方程:(sp[p]'=sp[p]+sp[p+S]=sum sp[k]T[k][p])。我们只令(T[p][p])(T[p+S][p])等于1就可以了。

    转移到1结尾的方程:(sp[p+S]'=sum_{k=0}^p sp[k]dbinom{p}{k}=sum_{k=0}^p sp[k]T[k][p+S])。 令(T[k][p+S]=dbinom{p}{k})就可以了。

    初状态:序列长为0。除了(sp[0]=0^0)外,其余都为0。末状态(sp'=sp cdot T^n)

    用线性递推求组合数时初始化为0,则(dbinom{p}{k})(k > p)时均为0,符合要求。

    总代码如下:

    #include<cstdio>
    #include<cstring>
    #include<cctype>
    #define RP(i,a,b) for(register int i=a;i<=b;++i)
    #define DRP(i,a,b) for(register int i=a;i>=b;--i)
    #define fre(z) freopen(z".in","r",stdin),freopen(z".out","w",stdout)
    #define mem(a,b) memset(a,b,sizeof(a))
    #define prique priority_queue
    #define rg register
    #include<cstdlib>
    #include<ctime>
    #ifdef WIN32
    #define OT "%I64d"
    #else
    #define OT "%lld"
    #endif
    using namespace std;
    typedef long long ll;
    typedef double db;
    template<class T> T qr(T type)
    {
        char ch=getchar();T ret=0;T q=1,res=1;
        while(!isdigit(ch))
            q=(ch=='-'?-1:q),ch=getchar();
        while(isdigit(ch))
            ret=ret*10+ch-'0',ch=getchar();
        if(ch=='.')
        {
            ch=getchar();
            while(isdigit(ch))
                ret=ret*10+ch-'0',res=res*10,ch=getchar();
        }
        return q==-1?-ret/res:ret/res;
    }
    
    const int maxn=10000002;
    const int maxa=50,maxm=100000000;
    const int maxS=100;
    
    ll n;
    int a,b,S;
    ll mod;
    
    struct mat{
        int n;
        ll v[maxS<<1][maxS<<1];
        mat()
        {
            memset(v,0,sizeof(v));
        }
    };
    
    inline mat mul(mat a,mat b)
    {
        mat ans;
        ans.n=a.n;
        RP(i,0,a.n-1)
        RP(j,0,b.n-1)
        {
            ll sum=0;
            RP(k,0,a.n-1)
            {
                sum+=(a.v[i][k])*b.v[k][j];
                if(sum>=mod)sum%=mod;
            }
            ans.v[i][j]=sum;
        }
        return ans;
    }
    
    ll C[maxS<<1][maxS<<1];
    ll np[maxa];
    
    inline void init()
    {
        np[0]=1;
        RP(i,1,a)
            np[i]=np[i-1]*n%mod;
        
        C[0][0]=1;
        RP(i,1,S-1)
        {
            C[i][0]=1;
            RP(j,1,i)
            C[i][j]=(C[i-1][j]+C[i-1][j-1])%mod;
        }
    }
    
    inline mat qp(mat s,ll p)
    {
        mat ans;
        ans.n=s.n;
        RP(i,0,s.n-1)
            ans.v[i][i]=1;
        while(p)
        {
            if(p&1)
                ans=mul(ans,s);
            s=mul(s,s);
            p>>=1;
        }
        return ans;	
    }
    
    inline ll f(ll k)
    {
        ll sgn=(a-k)&1?-1:1;
        return sgn*C[a][k]*np[k]%mod;
    }
    
    int main()
    {
        //fre("B");
        n=qr(1ll),a=qr(1),b=qr(1),mod=qr(1ll);
        S=a+b+1;
        init();
        
        mat T;
        T.n=S<<1;
        
        RP(i,0,S-1)
        {
            T.v[i][i]=T.v[i+S][i]=1;
            RP(j,i,S-1)
            T.v[i][j+S]=C[j][i];
        }
        T=qp(T,n);
        
        ll ans=0;
        RP(k,0,a)
        {
            ans=(ans+f(k)*(T.v[0][a+b-k]+T.v[0][a+b-k+S])%mod)%mod;
        }
        if(ans<0)
            ans=((ans%mod)+mod)%mod;
        printf("%lld",ans);
        return 0;
    }	
    
  • 相关阅读:
    属性与字段的区别
    修改LVDS支持1024*600分辨率
    Altium designer 10如何设置标题栏
    嵌入式C开发人员的最好的0x10道笔试题
    进程线程及堆栈关系的总结
    GDB调试
    c语言
    如何使用autotools生成Makefile
    ubuntu NFS
    Ubuntu安装配置TFTP服务
  • 原文地址:https://www.cnblogs.com/LinearODE/p/10626874.html
Copyright © 2011-2022 走看看