【HNOI】矩阵染色 数论

  【题目描述】一个2*i的矩阵,一共有m种颜色,相邻两个格子颜色不能相同,m种颜色不必都用上,f[i]表示这个答案,求Σf[i]*(2*i)^m (1<=i<=n)%p。

  【数据范围】

    20% n,m<10^5 p<10^9

    其余 n<10^9

      其中40% m<100 p<10^9

        20% m<10^3 p<10^9

        20% m<10^4 p<10^3

  首先我们可以推导出f[i]的式子,f[1]=m*(m-1),因为其余的格子,我们第一个格子可以放m-1种颜色,第二个格子在第一个格子和上一层第二个格子颜色不同时有m-2种情况,相同时有m-1种情况,那么我们可以得出f[i]=f[i-1]*(m^2-3*m+3),设a=m^2-3*m+3,那么问题就变成了求2^m*m*(m-1)/a*Σa^i*i^m (1<=i<=n)。

  对于前20%的数据,我们可以暴力的nlogm递推求解。

  对于中间的20%数据,我们可以化简一下求解式。

    设w[i][j]为Σa^j*j^k (1<=j<=i),那么答案就成了2^m*m*(m-1)/a*w[n][m],现在的问题就是求解w[n][m]。

    w[n][m]=Σa^i*i^m (1<=i<=n),w[n+1][m]=Σa^i*i^m (1<=i<=n+1)=a+Σa^i*i^m (2<=i<=n+1)=a+a*Σa^i*(i+1)^m (1<=i<=n)

    那么我们可以根据二项式展开来化简,w[n+1][m]=a+a*Σa^iΣc(m,j)*i^j (1<=i<=n) (0<=j<=m)=a+a*Σc(m,j)*w[n][j] (0<=j<=m),那么这样我们就可以写一个矩阵来加速在n^3logn的时间内求解了。

  对于m<10^3的情况,w[n+1][m]=w[n][m]+a^(n+1)*(i+1)^m。根据刚才的推导,w[n+1][m]=a+a*Σc(m,j)*w[n][j] (0<=j<=m)。所以我们可以得到

    w[n][m]+a^(n+1)*(i+1)^m=a+a*Σc(m,j)*w[n][j] (0<=j<=m),这样我们发现,如果我们知道了w[n][m]之前的w[n][j],那么我们可以在m的时间内反解出w[n][j]。

  对于最后的20%,我们发现p非常小,考虑答案的求和式Σa^i*i^m,发现这个式子之和i和i^m有关,当i,i^m和j,j^m关于mod p相等之后,那么i与j之后的变换是相同的,我们可以发现一共有p^2个不同的情况,那么我们记下来这个循环之后就可以算出来了。

//By BLADEVIL
#include <cstdio>
#define LL long long
#define maxp 1010
#define maxm 1010

using namespace std;

int n,m,p,a;
int mo[maxp],next[maxp],w[maxm],c[maxm][maxm];
int flag[maxp][maxp],f[maxp*maxp],g[maxp*maxp];

int mi(int a,int k) {
    int ans=1;
    while (k) {
        if (k&1) ans=((LL)ans*a)%p;
        a=((LL)a*a)%p;
        k>>=1;
    }
    return ans;
}

void work1() {
    int ans=0;
    ans=((LL)m*(m-1))%p; ans=((LL)ans*mi(2,m))%p; 
    ans=((LL)ans*mi(a,p-2))%p; //printf("%d\n",ans);
    int cur=0;
    for (int i=1;i<=n;i++) cur=((LL)cur+(LL)mi(a,i)*mi(i,m))%p;
    ans=((LL)ans*cur)%p;
    printf("%d\n",ans);
}

void work2() {
    int i,l,r; 
    f[1]=(LL)m*(m-1)%p; g[1]=(LL)f[1]*mi(2,m)%p;
    flag[1][f[1]]=1;
    for (i=2;i<=n;i++) {
        f[i]=(LL)f[i-1]*a%p;
        g[i]=(LL)f[i]*mi(2*i,m)%p;
        if (flag[i%p][f[i]]) {
            l=flag[i%p][f[i]];
            r=i-1;
            break;
        }
        flag[i%p][f[i]]=i;
    }
    //printf("%d %d %d %d\n",l,r,i,g[i]);
    int len=r-l+1,ans=0,tmp=0;
    for (int j=1;j<l;j++) ans=(ans+g[j])%p;
    n-=l-1;
    for (int j=l;j<=r;j++) tmp=(tmp+g[j])%p;
    ans=(ans+(LL)tmp*(n/len)%p)%p;
    for (int j=1;j<=n%len;j++) ans=(ans+g[l+j-1])%p;
    printf("%d\n",ans);
}

void work3() {
    for (int i=1;i<=m;i++) {
        c[i][0]=c[i][i]=1;
        for (int j=1;j<i;j++) c[i][j]=(c[i-1][j]+c[i-1][j-1])%p;
    }
    int ans=0;
    ans=((LL)m*(m-1))%p; ans=((LL)ans*mi(2,m))%p; 
    ans=((LL)ans*mi(a,p-2))%p; //printf("%d\n",ans);
    //for (int i=1;i<=m;i++) printf("%d ",c[m][i]); printf("\n");
    w[0]=((LL)mi(a,n+1)-a+p)%p; w[0]=((LL)w[0]*mi(a-1,p-2))%p;
    //printf("%d\n",w[0]);
    for (int i=1;i<=m;i++) {
        w[i]=((LL)mi(a,n+1)*mi(n+1,i)-a+p)%p;
        for (int j=0;j<i;j++) w[i]=((LL)w[i]-((LL)a*c[i][j]%p*w[j]%p)+p)%p;
        w[i]=(LL)w[i]*mi(a-1,p-2)%p;
        w[i]%=p;
    }
    //printf("%d\n",w[m]);
    ans=((LL)ans*w[m])%p;
    printf("%d\n",ans);
}

int main() {
    freopen("color.in","r",stdin); freopen("color.out","w",stdout);
    scanf("%d%d%d",&n,&m,&p);
    a=((LL)m*m-3*m+3)%p;
    if (n<=100000) work1(); else 
    if (m>1000) work2(); else work3();
    fclose(stdin); fclose(stdout);
    return 0;
}

 

posted on 2014-03-25 13:48  BLADEVIL  阅读(661)  评论(0编辑  收藏  举报