POJ 3070 Fibonacci (初学矩阵快速幂)

按往常一样,记下一些好的资料:

http://www.matrix67.com/blog/archives/276 这是矩阵乘法10个经典的题目,其中最后一题的最后一段在该文的评论中给出。


矩阵乘法满足结合律保证了矩阵快速幂的正确性。

目前的代码风格是学了九野的模版风格(http://blog.csdn.net/acmmmm/article/details/10041141

#define Matr 10 //矩阵大小,注意能小就小

struct mat//矩阵结构体,a表示内容,size大小 矩阵从1开始
{
    ll a[Matr][Matr],size;
    mat()
    {
        size=0;
        memset(a,0,sizeof(a));
    }
};
void print(mat m)//输出矩阵信息,debug用 
{
    int i,j;
    printf("%d\n",m.size);
    for(i=0;i<m.size;i++)
    {
        for(j=0;j<m.size;j++)printf("%d ",m.a[i][j]);
        printf("\n");
    }
}

mat multi(mat m1,mat m2,int mod)//两个相等矩阵的乘法,对于稀疏矩阵,有0处不用运算的优化 
{
    mat ans=mat();    ans.size=m1.size;
    for(int i=1;i<=m1.size;i++)
        for(int j=1;j<=m2.size;j++)
            if(m1.a[i][j])//稀疏矩阵优化 
                for(int k=1;k<=m1.size;k++)
                    ans.a[i][k]=(ans.a[i][k]+m1.a[i][j]*m2.a[j][k])%mod;

    return ans;
}
mat quickmulti(mat m,int n,int mod)//二分快速幂 
{
    mat ans=mat();
    int i;
    for(i=1;i<=m.size;i++)ans.a[i][i]=1;
    ans.size=m.size;
    while(n)
    {
        if(n&1)ans=multi(m,ans,mod);
        m=multi(m,m,mod);
        n>>=1;
    }
    return ans;
}
/*
ans^=n ->
mat ans=mat();
ans.size=Size;
初始化ans矩阵
ans=quickmulti(ans,n,mod);
*/


  
模版中矩阵乘法不按照矩阵乘法定义的顺序来进行是为了稀疏矩阵的优化,按找定义执行那么条件判断会变多,效率会低

对于本题考察了:

任何一个线性递推式的第n项,其对应矩阵的构造方法为:在右上角的(n-1)*(n-1)的小矩阵中的主对角线上填1,矩阵第n行填对应的系数,其它地方都填0。


题目大意就是求出斐波那契数列第k项的后四位,如果有前导0就不输出0,但最后一个0必须输出。

#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
#define mod 10000
int n;
struct mat{
    int a[3][3];
    void ini(){
        memset(a,0,sizeof(a));
    }
};

mat multi(mat m1,mat m2){
    mat ans;
    ans.ini();
    for(int i=1;i<=2;i++)
        for(int j=1;j<=2;j++)
            for(int k=1;k<=2;k++)
                ans.a[i][k]=(ans.a[i][k]+m1.a[i][j]*m2.a[j][k])%mod;
    return ans;
}
mat quickmulti(mat m,int n){
    mat ans;
    ans.ini();
    for(int i=1;i<=2;i++) ans.a[i][i]=1;
    while(n){
        if(n&1){
            ans=multi(ans,m);
        }
        m=multi(m,m);
        n>>=1;
    }
    return ans;
}
int main(){
    while(scanf("%d",&n),~n){
        mat m;
        m.a[1][1]=1;
        m.a[1][2]=1;
        m.a[2][1]=1;
        m.a[2][2]=0;
        mat ans=quickmulti(m,n);
        printf("%d\n",ans.a[1][2]);
    }
    return 0;
}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值