[AT2384] [agc015_f] Kenus the Ancient Greek

題目連接

AtCoder:https://agc015.contest.atcoder.jp/tasks/agc015_fc++

洛谷:https://www.luogu.org/problemnew/show/AT2384git

Solution

神仙結論題...窩只會打表找規律...ide

咱們定義\(f(i,j)\)表示\((i,j)\)\(\rm Euclidean\ step\ count\),也就是走多少步那個玩意。idea

定義\(Fib(n)​\)表示斐波那契數列第\(n​\)項,其中\(Fib(0)=Fib(1)=1​\)spa

有一個比較顯然(好找出規律)的結論:\(f(Fib(x),Fib(x+1))=x​\),且不存在任意\((i,j)​\)知足\(f(i,j)\geqslant x,i< Fib(x),j< Fib(x+1)​\),這個由(打表)數學概括可得。code

咱們定義一個二元組\((x,y)​\)是好的,當且僅當不存在\((i,j)​\)知足\(i<x,j<y,f(i,j)>f(x,y)​\),顯然只有好的二元組能貢獻答案。get

咱們定義一個二元組\((x,y)​\)是優秀的,當且僅當\(x,y\leqslant Fib(v+2)+Fib(v-1)​\),其中\(v=f(x,y)​\)數學

那麼有一個結論:一個好的二元組進行一次\(\rm Euclidean\ step​\)以後必定爲一個優秀的二元組,證實以下:it

咱們考慮反證法證實,設好的二元組爲\((a,b)=(y,py+x)\)知足\(x\leqslant y,f(x,y)=v+1\),優秀的二元組爲\((x,y)\),假設\(y> Fib(v+2)+Fib(v-1)\)io

可得:\(a=y>Fib(v+2),b=py+x\geqslant x+y>Fib(v)+F(v+2)+Fib(v-1)=Fib(v+3)\)

注意到\(f(Fib(v+2),Fib(v+3))=v+2>f(a,b)\),即存在\((a',b')\)知足\(a'>a,b'>b\)\(f(a',b')>f(a,b)\),與\((a,b)\)是好的二元組矛盾。

打表可知優秀的二元組並很少,貌似是\(O(\log^2 n)​\)級別?我也不是特別清楚,因此咱們能夠把全部的優秀的二元組預處理出來,而後求答案就行了。

複雜度多是\(O(q\log n)​\)...

#include<bits/stdc++.h>
using namespace std;

#define int long long 

void read(int &x) {
    x=0;int f=1;char ch=getchar();
    for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
    for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
}

void print(int x) {
    if(x<0) putchar('-'),x=-x;
    if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');}

#define lf double
#define ll long long 

const int maxn = 110;
const int inf = 1e9;
const lf eps = 1e-8;
const int mod = 1e9+7;

#define pii pair<int,int >
#define fr first
#define sc second

#define vec vector<pii >
#define mp make_pair
#define pb push_back

#define iter vector <pii > :: iterator

vec t[maxn];
int n,m,q,f[maxn];

signed main() {
    t[1].pb(mp(1,2)),t[1].pb(mp(1,3)),t[1].pb(mp(1,4));
    f[0]=f[1]=1;for(int i=2;i<=100;i++) f[i]=f[i-1]+f[i-2];
    for(int i=1;i<=100;i++) 
        for(int j=0,l=t[i].size()-1;j<=l;j++) {
            int x=t[i][j].sc,y=t[i][j].fr+x;
            while(y<=f[i+3]+f[i]) t[i+1].pb(mp(x,y)),y+=x;
        }read(q);
    for(int i=1;i<=q;i++) {
        read(n),read(m);if(n>m) swap(n,m);
        int p=1,ans=0;
        while(f[p+1]<=n&&f[p+2]<=m) p++;
        printf("%lld ",p);
        if(p==1) {write(n%mod*m%mod);continue;}
        for(int j=0,l=t[p-1].size()-1;j<=l;j++) {
            int x=t[p-1][j].fr,y=t[p-1][j].sc;
            if(y<=n) ans=(ans+(m-x)/y)%mod;
            if(y<=m) ans=(ans+(n-x)/y)%mod;
        }write(ans);
    }
    return 0;
}
相關文章
相關標籤/搜索