hdu 3117 Fibonacci Numbers 矩阵快速幂+公式

斐波那契数列后四位可以用快速幂取模(模10000)算出。前四位要用公式推

HDU 3117 Fibonacci Numbers(矩阵快速幂+公式)

f(n)=(((1+√5)/2)^n+((1-√5)/2)^n)/√5

假设F[n]可以表示成 t * 10^k(t是一个小数),那么对于F[n]取对数log10,答案就为log10 t + K,此时很明显log10 t<1,于是我们去除整数部分,就得到了log10

再用pow(10,log10 t)我们就还原回了t。将t×1000就得到了F[n]的前四位。 具体实现的时候Log10 F[n]约等于((1+√5)/2)^n/√5,这里我们把((1-√5)/2)^n这一项忽略了,
因为当N>=40时,这个数已经小的可以忽略。于是log10 F[n]就可以化简成log10 1/√5 + n*log10 (1+√5)/2

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <cmath>
#include <cstring>
using namespace std;

const int Mod = 10000;
const int N = 4;
int msize;

struct Mat
{
    int mat[N][N];
};

Mat operator *(Mat a, Mat b)
{
    Mat c;
    memset(c.mat, 0, sizeof(c.mat));
    for(int k = 0; k < msize; ++k)
        for(int i = 0; i < msize; ++i)
            if(a.mat[i][k])
                for(int j = 0; j < msize; ++j)
                    if(b.mat[k][j])
                        c.mat[i][j] = (c.mat[i][j] +a.mat[i][k] * b.mat[k][j])%Mod;
    return c;
}

Mat operator ^(Mat a, int k)
{
    Mat c;
    memset(c.mat,0,sizeof(c.mat));
    for(int i = 0; i < msize; ++i)
        c.mat[i][i]=1;
    for(; k; k >>= 1)
    {
        if(k&1) c = c*a;
        a = a*a;
    }
    return c;
}

int a[50];
const double sq5 = sqrt(5.0);

int main()
{
//    freopen("in.txt","r",stdin);
    int n;
    msize = 2;
    Mat A;
    A.mat[0][0] = 1;
    A.mat[0][1] = 1;
    A.mat[1][0] = 1;
    A.mat[1][1] = 0;
    a[0] = 0;
    a[1] = 1;
    for(int i=2;i<40;i++)
        a[i] = a[i-1] + a[i-2];
    while(~scanf("%d", &n))
    {
        if(n<40) printf("%d\n",a[n]);
        else
        {
            double t = n * log10((1.0 + sq5) / 2) - log10(sq5);
            t -= (int)t;
            t = pow(10.0, t);
            printf("%d...",(int)(t * 1000));
            Mat aa = A;
            aa = aa ^ (n-1);
            printf("%04d\n",aa.mat[0][0]);
        }
    }
    return 0;
}

 

posted @ 2017-07-24 21:03  Pacify  阅读(188)  评论(0编辑  收藏  举报