• hdu 5451 Best Solver(矩阵快速幂+循环节)


    题意:

    已知,给你整数x,和一个素数M,求[y]%M

    思路:

    (5+2√6)n=Xn+Yn*√6

    Xn+Yn*√6 =(Xn-1+Yn-1*√6)*(5+2√6)

                     => 5*Xn-1 + 12*Yn-1 + (2*Xn-1 + 5*Yn-1 )*√6

    Xn =  5*Xn-1 + 12*Yn-1;

    Y=  2*Xn-1 + 5*Yn-1;

    然而√6还是一个大问题,解决办法:

    (5 - 2√6)= Xn - Yn*√6

    (5+2√6)n=Xn+Yn*√6 + X- Yn*√6 - (X- Yn*√6) = 2*Xn - (0.101)n

    所以[ (5+2√6)n ]= 2*X- 1

    由于n=1+2^x 是一个非常大的数,这时我们要知道广义斐波那契数列的循环节是(MOD-1)*(MOD+1)

    代码:

    #include <cstdio>
    #include <iostream>
    #include <cmath>
    #include <algorithm>
    #define ll long long
    
    using namespace std;
    
    const int N=2,M=2,P=2;
    ll MOD;
    
    struct Matrix
    {
        ll m[N][N];
    };
    
    
    Matrix A= {5,12,
               2,5
              };
    
    Matrix I= {1,0,
               0,1
              };
    
    Matrix multi(Matrix a,Matrix b)
    {
        Matrix ans;
        for(int i=0; i<N; i++)
        {
            for(int j=0; j<M; j++)
            {
                ans.m[i][j]=0;
                for(int k=0; k<P; k++)
                {
                    ans.m[i][j]+=a.m[i][k]*b.m[k][j]%MOD;
                }
                ans.m[i][j]%=MOD;
            }
        }
        return ans;
    }
    
    Matrix power(Matrix a,ll k)
    {
        Matrix ans=I,p=a;
        while(k)
        {
            if(k&1)
            {
                ans=multi(ans,p);
            }
            k=k/2;
            p=multi(p,p);
        }
        return ans;
    }
    
    ll qmod(ll a,ll b,ll mod)
    {
        ll ans=1;
        a=a%mod;
        while(b)
        {
            if(b&1)
            {
                ans=(ans*a)%mod;
            }
            b=b/2;
            a=(a*a)%mod;
        }
        return ans;
    }
    
    int main(int argc, char const *argv[])
    {
        int t,cas=1,n;
        cin>>t;
        while(t--)
        {
            cin>>n>>MOD;
            cout<<"Case #"<<cas++<<": ";
            n=qmod(2,n,(MOD*MOD-1))+1;
            Matrix ans=power(A,n-1);
            ll xn=(5*ans.m[0][0]+2*ans.m[0][1])%MOD;
            cout<<(2*xn-1)%MOD<<endl;
        }
        return 0;
    }
  • 相关阅读:
    四组API
    常用的辅助类(必会
    时间戳
    SpringMVC JSON乱码解决
    数据显示到前端
    ubuntu vim字体高亮
    vi编辑文件保存后,提示""vimrc" E212: Can't open file for writing Press ENTER or type command to continu"
    C语言程序设计100例之(25):确定进制
    C语言程序设计100例之(24):数制转换
    C语言程序设计100例之(23):数列求和
  • 原文地址:https://www.cnblogs.com/simplekinght/p/7074606.html
Copyright © 2020-2023  润新知