【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;

}

 

你可能感兴趣的:(矩阵)