poj 3735 大数量反复操作问题(矩阵高速幂)

题意:一个一维数组,3种操作: a:  第i个数+1,b: 第i个数=0 ,c::交换某俩处的数。  由三种基本操作构成一组序列,反复该序列m次(m<10^9),问结果

属于一种综合操作反复型: 每次乘以一矩阵T,相当于做一次操作。关键是构造这个矩阵。

先构造最初矩阵A :  1*(n +1) ={1,0,0,0...} ,  第一个一时为了操作第一行数的,   

 T的构造:初始是N+1 * N+1单位阵 这样恰好操作第i个数, +1,就在第0行的第 i个加1;交换就相应列交换,清零就相应列清0.

ans= A*(T^m); 注意用;long long 

#include<iostream>
#include<cstring>
using namespace std;
struct juz
{
    long long  bat[105][105];
    int x,y;     //行 列
    juz ()
    {
        memset(bat,0,sizeof(bat));
        x=0;y=0;
    }
};
juz mutp(juz a,juz b)
{
    juz c;
    c.x=a.x;c.y=b.y;
    memset(c.bat,0,sizeof(c.bat));
    for(int k=0;k<a.y;k++)
          for(int i=0;i<a.x;i++)
          if(a.bat[i][k])
          {
              for(int j=0;j<b.y;j++)
              {
                  c.bat[i][j]+=(a.bat[i][k]*b.bat[k][j]);
              }
          }
    return c;
}
juz quickf(juz a,int k)
{
    juz c=a;
    for(int i=0;i<a.x;i++)
      for(int j=0;j<a.x;j++)
          c.bat[i][j]=(i==j);
    while(k>=1)
    {
        if(k%2)
            c=mutp(c,a);
        k=k/2; a=mutp(a,a);
    }
    return c;
}
int main()
{
    int n,m,k;
    while(cin>>n>>m>>k&&(n||m||k))
    {
        juz a,b,c;
        a.x=1;a.y=n+1; b.x=n+1;b.y=n+1;
        for(int i=0;i<=n;i++)
        {
            a.bat[0][i]=0;
            b.bat[i][i]=1;
        }
        a.bat[0][0]=1;
        char tmp;
        int xx,yy;
        for(int i=0;i<k;i++)
        {
            cin>>tmp;
            if(tmp=='g')
            {
                cin>>xx;
                b.bat[0][xx]++;
            }
            else if(tmp=='e')
            {
                cin>>xx;
                for(int i=0;i<=n;i++)
                  b.bat[i][xx]=0;
            }
            else
            {
                cin>>xx>>yy;
                for(int i=0;i<=n;i++)
                {
                    int tx=b.bat[i][xx];
                    b.bat[i][xx]=b.bat[i][yy];
                    b.bat[i][yy]=tx;
                }
            }
        }
        c=quickf(b,m);
        c=mutp(a,c);
       for(int i=1;i<=n;i++)
         if(i!=n)cout<<c.bat[0][i]<<" ";
         else cout<<c.bat[0][i]<<endl;
    }
    return 0;
}





posted @ 2014-10-13 09:08  blfshiye  阅读(128)  评论(0编辑  收藏  举报