@description@

Fib数列为1,1,2,3,5,8...

求在Mod10^9+9的意义下,数字N在Fib数列中出现在哪个位置

无解输出-1

原题传送门。

@solution@

一个熟练的 OIer 选手应该能迅速发现 5 在模 10^9 + 9 意义下有二次剩余。考虑斐波那契通项公式:

\[f_i = \frac{1}{\sqrt{5}}((\frac{1 + \sqrt{5}}{2})^n - (\frac{1 - \sqrt{5}}{2})^n) = N
\]

不妨记 \(a = \frac{1 + \sqrt{5}}{2}, b = \frac{1 - \sqrt{5}}{2}\),则我们要解方程 \(a^n - b^n = \sqrt{5}N = K\)。

这个方程可解吗?注意 a, b 存在关系:\(a + b = 1, ab = -1\)。前一个显然不好用,我们用后一个:

\[a^n - (\frac{-1}{a})^n = K\\
a^{2n} - Ka^{n} - (-1)^n = 0\\
a^n = \frac{K \pm \sqrt{K^2 + 4\times(-1)^n}}{2}
\]

讨论一下 n 的奇偶可以解出 \(a^n\),然后 BSGS 即可。平方根也可 BSGS + 原根来解。

@accepted code@

#include <cstdio>
#include <vector>
#include <cassert>
#include <iostream>
#include <algorithm>
using namespace std; const int MOD = int(1E9) + 9;
const int INV2 = (MOD + 1) / 2;
const int INV5 = (MOD + 1) / 5;
const int SQ5 = 383008016;
const int A = 1LL*(MOD + 1 + SQ5)*INV2%MOD;
const int B = 1LL*(MOD + 1 - SQ5)*INV2%MOD;
const int C = 1LL*SQ5*INV5%MOD;
const int G = 13;
const int BLOCK = 32000;
const int HASHSIZE = 1000037;
const int MA = 133086171;
const int INVMA = 74832817;
const int GCD = 3; inline int add(int x, int y) {return (x + y >= MOD ? x + y - MOD : x + y);}
inline int sub(int x, int y) {return (x - y < 0 ? x - y + MOD : x - y);}
inline int mul(int x, int y) {return 1LL * x * y % MOD;} int pow_mod(int b, int p) {
int ret = 1;
for(int i=p;i;i>>=1,b=mul(b,b))
if( i & 1 ) ret = mul(ret, b);
return ret;
}
int fib(int n) {
int x = sub(pow_mod(A, n), pow_mod(B, n));
return mul(C, x);
} vector<pair<int, int> >h[HASHSIZE];
int hash_search(int x) {
int y = x % HASHSIZE;
for(int i=0;i<h[y].size();i++)
if( h[y][i].first == x )
return h[y][i].second;
return -1;
}
void hash_insert(int x, int k) {
int y = x % HASHSIZE;
for(int i=0;i<h[y].size();i++)
if( h[y][i].first == x )
return ;
h[y].push_back(make_pair(x, k));
}
void init() {
int p = pow_mod(G, BLOCK);
for(int i=1,q=p;i<=BLOCK;i++,q=mul(q,p))
hash_insert(q, i*BLOCK);
}
int bsgs(int x) {
if( x == 0 ) return -1;
int ret = MOD;
for(int i=1,q=G;i<=BLOCK;i++,q=mul(q,G)) {
int p = hash_search(mul(x, q));
if( p != -1 ) ret = min(ret, p - i);
}
assert(pow_mod(G, ret) == x);
return ret;
}
int msqrt(int x) {
if( x == 0 ) return 0;
int p = bsgs(x);
return p % 2 ? -1 : pow_mod(G, p / 2);
} int ans = -1;
void update(int x, int r) {
x = 1LL*(x / GCD)*INVMA%((MOD - 1) / GCD);
if( x % 2 == r )
if( ans == -1 || ans > x )
ans = x;
}
int main() {
int N, K; scanf("%d", &N), K = mul(N, SQ5), init(); int P = msqrt(add(mul(K, K), 4)); // even
if( P != -1 ) {
int k = bsgs(mul(add(K, P), INV2));
if( k != -1 && k % GCD == 0 ) update(k, 0);
k = bsgs(mul(sub(K, P), INV2));
if( k != -1 && k % GCD == 0 ) update(k, 0);
} P = msqrt(sub(mul(K, K), 4)); // odd
if( P != -1 ) {
int k = bsgs(mul(add(K, P), INV2));
if( k != -1 && k % GCD == 0 ) update(k, 1);
k = bsgs(mul(sub(K, P), INV2));
if( k != -1 && k % GCD == 0 ) update(k, 1);
} if( ans != -1 ) assert(fib(ans) == N);
printf("%d\n", ans);
}

@details@

因为几乎都是常数,可以预处理出来直接用。

话说这道题的主要难点是解方程那一块吧,感觉数学味儿多一点。

@bzoj - 5104@ Fib数列的更多相关文章

  1. BZOJ 5104 Fib数列(二次剩余+BSGS)

    斐波那契数列的通项: \[\frac{1}{\sqrt{5}}((\frac{1+\sqrt{5}}{2})-(\frac{1-\sqrt{5}}{2}))\] 设T=\(\sqrt{5}*N\),\ ...

  2. 【BZOJ5104】Fib数列(BSGS,二次剩余)

    [BZOJ5104]Fib数列(BSGS,二次剩余) 题面 BZOJ 题解 首先求出斐波那契数列的通项: 令\(A=\frac{1+\sqrt 5}{2},B=\frac{1-\sqrt 5}{2}\ ...

  3. FIB数列

    斐波那契级数除以N会出现循环,此周期称为皮萨诺周期. 下面给出证明 必然会出现循环 这是基于下面事实: 1. R(n+2)=F(n+2) mod P=(F(n+1)+F(n)) mod P=(F(n+ ...

  4. bzoj5104: Fib数列

    Description Fib数列为1,1,2,3,5,8... 求在Mod10^9+9的意义下,数字N在Fib数列中出现在哪个位置 无解输出-1 Input 一行,一个数字N,N < = 10 ...

  5. 动态规划之Fib数列类问题应用

    一,问题描述 有个小孩上楼梯,共有N阶楼梯,小孩一次可以上1阶,2阶或者3阶.走到N阶楼梯,一共有多少种走法? 二,问题分析 DP之自顶向下分析方式: 爬到第N阶楼梯,一共只有三种情况(全划分,加法原 ...

  6. UVaLive 3357 Pinary (Fib数列+递归)

    题意:求第 k 个不含前导 0 和连续 1 的二进制串. 析:1,10,100,101,1000,...很容易发现长度为 i 的二进制串的个数正好就是Fib数列的第 i 个数,因为第 i 个也有子问题 ...

  7. 【bzoj5118】Fib数列2 费马小定理+矩阵乘法

    题目描述 Fib定义为Fib(0)=0,Fib(1)=1,对于n≥2,Fib(n)=Fib(n-1)+Fib(n-2) 现给出N,求Fib(2^n). 输入 本题有多组数据.第一行一个整数T,表示数据 ...

  8. HDU3977 Evil teacher 求fib数列模p的最小循环节

    In the math class, the evil teacher gave you one unprecedented problem! Here f(n) is the n-th fibona ...

  9. 1022. Fib数列

    https://acm.sjtu.edu.cn/OnlineJudge/problem/1022 Description 定义Fib数列:1,1,2,3,5,8,13,…1,1,2,3,5,8,13, ...

随机推荐

  1. mysql小白系列_09 mysql性能优化关键点

    一 服务器参数调优,有哪些关键点? 1. 应用访问优化 优化方法 性能提升效果 优化成本 说明 减少数据访问能不访问就不访问-减少磁盘IO 1~1000 低 缓存服务器缓存mysql数据,Redis. ...

  2. HDU 6047 贪心思维题

    Maximum Sequence Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  3. 花费一周刷完两份面试pdf(含答案)轻松拿下了抖音、头条、京东、小米等大厂的offer,成功度过程序员的寒冬。

    整理出一篇Java进阶架构师之路的核心知识,同时也是面试时面试官必问的知识点,篇章也是包括了很多知识点,其中包括了有基础知识.Java集合.JVM.多线程并发.spring原理.微服务.Netty 与 ...

  4. 《机器学习Python实现_09_02_决策树_CART》

    简介 CART树即分类回归树(classification and regression tree),顾名思义,它即能用作分类任务又能用作回归任务,它的应用比较广泛,通常会用作集成学习的基分类器,总得 ...

  5. Golang源码学习:调度逻辑(三)工作线程的执行流程与调度循环

    本文内容主要分为三部分: main goroutine 的调度运行 非 main goroutine 的退出流程 工作线程的执行流程与调度循环. main goroutine 的调度运行 runtim ...

  6. 5.List链表类型介绍和操作

    数据类型List链表 (1)介绍 list类型其实就是一个双向链表.通过push,pop操作从链表的头部或者尾部添加删除元素.这使得list既可以用作栈,也可以用作队列. 该list链表类型应用场景: ...

  7. 华为五大专家亲述:如何转型搞 AI?

    导语:非AI专业技术人员转型AI技术,或是作为一名学生学习AI技术开发,对每个有这样诉求和经历的人来说,都希望能够看到AI技术人才的成长经历,给出自己的真实经历分享. 前言 参考塞缪尔.约翰逊(18世 ...

  8. jchdl - GSL实例 - Mux4(使用WireVec简化输入线声明)

    https://mp.weixin.qq.com/s/yJx_dV6ScUStJtPWVuD38w 原理图 ​​ 参考链接 https://github.com/wjcdx/jchdl/blob/ma ...

  9. DDD之3实体和值对象

    图中是一个别墅的模型,代表实体,可以真实的看得到.那么在DDD设计方法论中,实体和值对象是什么呢? 背景 实体和值对象是领域模型中的领域对象,是组成领域模型的基础单元,一起实现实体最基本的核心领域逻辑 ...

  10. Java实现 LeetCode 461 汉明距离

    461. 汉明距离 两个整数之间的汉明距离指的是这两个数字对应二进制位不同的位置的数目. 给出两个整数 x 和 y,计算它们之间的汉明距离. 注意: 0 ≤ x, y < 231. 示例: 输入 ...