2016 pku campus/OpenJ_POJ - C16H(推公式+矩阵快速幂)
描述
Wenwen has a magical ball. When put on an infinite plane, it will keep duplicating itself forever.
Initially, Wenwen puts the ball on the location (x0, y0) of the plane. Then the ball starts to duplicate itself right away. For every unit of time, each existing ball on the plane will duplicate itself, and the new balls will be put on the adjacent locations. The duplication rule of these balls is, during the i-th unit of time, a ball, which locates at (x, y), will duplicate ui balls to (x, y+1), di balls to (x, y-1), li balls to (x-1, y) and ri balls to (x+1, y).
The duplication rule has a period of M. In another words, ui=ui-M, di=di-M, li=li-M, ri=ri-M, for i=M+1,M+2,...
Wenwen is very happy because she will get many balls. It is easy to calculate how many balls she will get after N units of time. However, she wants to know the sum of x-coordinates and y-coordinates of all balls after N units of time. This is a bit difficult for her. Could you help her? Since the sum might be very large, you should give the sum modulo 1,000,000,007 to her.
输入The first line contains an integer T (1 ≤ T ≤ 25), indicating the number of test cases.For each test case:
The first line contains four integers N (1 ≤ N ≤ 10^18), M (1 ≤ M ≤ 20,000), x0 and y0 (-10^18 ≤ x0,y0 ≤ 10^18);
Then follows M lines, the i-th line contains four integers: ui, di, li and ri (0 ≤ ui,di,li,ri ≤ 10,000).输出For each test case, output one integer on a single line, indicating the sum of x-coordinates and y-coordinates of all balls after N units of time, modulo 1,000,000,007.样例输入
1 2 2 1 1 2 0 0 0 0 0 0 1样例输出
19
Initially, there is 1 ball on (1,1).
After 1 unit of time, there is 1 ball on (1,1) and 2 balls on (1,2);
After 2 units of time, there is 1 ball on (1,1), 2 balls on (1,2), 1 ball on (2,1) and 2 balls on (2,2).
Therefore, after 2 units of time, the sum of x-coordinates and y-coordinates of all balls is
(1+1)*1+(1+2)*2+(2+1)*1+(2+2)*2=19.
#include <bits/stdc++.h>
#define maxn 100005
using namespace std;
const int mod=1e9+7;
typedef long long ll;
struct Matrix{
    ll mo[2][2];
    Matrix(){
        memset(mo,0,sizeof(mo));
    }
};
Matrix Mul(Matrix x,Matrix y){
    Matrix c;
    for(int i=0;i<2;i++){
        for(int j=0;j<2;j++){
            for(int k=0;k<2;k++){
                c.mo[i][j]=(c.mo[i][j]+x.mo[i][k]*y.mo[k][j])%mod;
            }
        }
    }
    return c;
}
Matrix powmod(Matrix x,ll n){
    Matrix res;
    for(int i=0;i<2;i++){
        res.mo[i][i]=1;
    }
    while(n){
        if(n&1) res=Mul(res,x);
        n>>=1;
        x=Mul(x,x);
    }
    return res;
}
int A[maxn];
int B[maxn];
int main()
{
    int t;
    scanf("%d",&t);
    while(t--){
        ll n,m,x0,y0;
        scanf("%lld%lld%lld%lld",&n,&m,&x0,&y0);
        x0=(x0%mod+mod)%mod;
        y0=(y0%mod+mod)%mod;
        for(int i=0;i<m;i++){
            int a,b,c,d;
            scanf("%d%d%d%d",&a,&b,&c,&d);
            A[i]=1+a+b+c+d;
            B[i]=a-b+d-c;
        }
        Matrix base;
        for(int i=0;i<2;i++){
            base.mo[i][i]=1;
        }
        for(int i=0;i<m;i++){
            Matrix tmp;
            tmp.mo[0][0]=tmp.mo[1][1]=A[i],tmp.mo[1][0]=B[i];
            base=Mul(base,tmp);
        }
        base=powmod(base,n/m);
        for(int i=0;i<n%m;i++){
            Matrix tmp;
            tmp.mo[0][0]=tmp.mo[1][1]=A[i],tmp.mo[1][0]=B[i];
            base=Mul(base,tmp);
        }
        cout<<(base.mo[1][0]+base.mo[1][1]*(x0+y0))%mod<<endl;
    }
    return 0;
}

 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号