P1306 斐波那契公约数 题解

请求出 fnfm 的最大公约数,即 gcd(fn,fm),答案对 108 取模。

结论:gcd(fn,fm)=fgcd(n,m)

证明如下:

首先引理 1:

fn+m=fn1×fm+fn×fm+1

运用归纳法,可以简单证明,此处略去。

引理 2:

gcd(fn,fn+1)=1

运用归纳法:

gcd(f1,f2)=1 平凡成立。

设现已有 gcd(fn1,fn)=1,则

gcd(fn,fn+1)=gcd(fn,fn1+fn)=gcd(fn1,fn)=1

证毕。

1fm=fnm1fn+fnmfn+1gcd(fn,fm)=gcd(fn,fnm1fn+fnmfn+1)fn|fnm1fngcd(fn,fm)=gcd(fn,fnmfn+1)2gcd(fn,fn+1)=1gcd(fn,fm)=gcd(fn,fnm)gcd(fn,fm)=gcd(fnm,fn)gcd(fn,fm)=gcd(fn mod m,fn)gcd(fn,fm)=gcd(fgcd(n,m),0)=fgcd(n,m)

最后用矩阵算 fgcd(n,m) 即可。

#include <algorithm>
#include <cstring>
#include <iostream>
#define speedup (ios::sync_with_stdio(0), cin.tie(0), cout.tie(0))
#define int long long
using namespace std;

const int mod = 1e8;

struct Mat
{
    int m[5][5];
    int r, c;
    void clear(int R, int C)
    {
        memset(m, 0, sizeof m);
        r = R, c = C;
    }
    void init()
    {
        for (int i = 0; i <= min(r, c); i++)
            m[i][i] = 1;
    }
    friend Mat operator*(const Mat a, const Mat b)
    {
        Mat res;
        res.clear(a.r, b.c);
        for (int i = 1; i <= a.r; i++)
            for (int j = 1; j <= b.c; j++)
                for (int k = 1; k <= a.c; k++)
                    res.m[i][j] = (res.m[i][j] + a.m[i][k] * b.m[k][j] % mod) % mod;
        return res;
    }
} M, A;

Mat qmi(Mat a, int b)
{
    Mat res;
    res.clear(3, 3), res.init();
    while (b)
    {
        if (b & 1)
            res = res * a;
        b >>= 1;
        a = a * a;
    }
    return res;
}

signed main()
{
    speedup;
    int n, m;
    cin >> n >> m;
    M.clear(2, 2);
    M.m[1][2] = M.m[2][1] = M.m[2][2] = 1;
    A.clear(2, 1);
    A.m[2][1] = 1;
    A = qmi(M, __gcd(n, m) - 1) * A;
    cout << A.m[2][1] << '\n';

    return 0;
}
posted @   MoyouSayuki  阅读(17)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
:name :name
点击右上角即可分享
微信分享提示