[zoj 3774]Power of Fibonacci 数论(二次剩余 拓展欧几里得 等比数列求和)
Power of Fibonacci
Time Limit: 5 Seconds Memory Limit: 65536 KB
In mathematics, Fibonacci numbers or Fibonacci series or Fibonacci sequence are the numbers of the following integer sequence:
1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233, 377, ...
By definition, the first two numbers in the Fibonacci sequence are 1 and 1, and each subsequent number is the sum of the previous two. In mathematical terms, the sequence Fn of
Fibonacci numbers is defined by the recurrence relation Fn = Fn - 1 + Fn - 2 with seed values F1 = 1 and F2 = 1.
And your task is to find ΣFiK, the sum of the K-th power of the first N terms in the Fibonacci sequence. Because the answer can
be very large, you should output the remainder of the answer divided by 1000000009.
Input
There are multiple test cases. The first line of input is an integer T indicates the number of test cases. For each test case:
There are two integers N and K (0 <= N <= 1018, 1 <= K <= 100000).
Output
For each test case, output the remainder of the answer divided by 1000000009.
Sample Input
5
10 1
4 20
20 2
9999 99
987654321987654321 98765
Sample Output
143
487832952
74049690
113297124
108672406
Hint
The first test case, 1 + 1 + 2 + 3 + 5 + 8 + 13 + 21 + 34 + 55 = 143.
The second test case, 120 + 120 + 220 + 320 =3487832979, and 3487832979 = 3 * 1000000009 + 487832952, so the output is 487832952.
题目大意
非常明白,求菲波那契数列每一个数的m次方的前n项和。
思路
当时看到m的取值范围就没敢在往下写。直到cf 255(446 C)有一道类似思路的线段树,官方的editorial里面给出了一个解释,就是求其二次剩余,然后该数列能够用幂差来表示。


(二次剩余求解)


(以上三个式子均用拓展欧几里得推出。将sqrt5作为一个总体进行逆元求解)
(于是带入后我们就看到了一个剩余系的式子)
为什么要这么做的原因可能就和逆元的原理有些类似了,由于我们直到结果是整数,所以每一步都能在剩余系之中找到一个整数的运算取代了。
接下来就设
于是随意一个数的m次方为
对二项式展开之后其前n项求和
枚举k来求和就可以
对于中间的一项等比数列的求和来加速。否则仍会超时。
求二次剩余以及各项系数值的code:(因为二次剩余大多有两个不同根。所以结果不同未必影响计算结果)
模板来自:blog.csdn.net/acdreamers/article/details/10182281
#include <cstdio>
#include <cstring>
#include <algorithm>
#define mod 1000000009
using namespace std;
long long sqrt5,s,r1,r2;
long long ts,w;
struct D{
long long p,d;
};
void egcd(long long a,long long b,long long &x,long long &y)
{
if (b==0)
{
x=1;
y=0;
return;
}
egcd(b,a%b,x,y);
int t=x;
x=y,y=t-a/b*y;
return;
}
long long mypow(long long x,long long y,long long p)
{
long long res=1,mul=x;
while (y)
{
if (y & 1)
res=res * mul % p;
mul=mul * mul % p;
y/=2;
}
return res;
}
D mul(D a,D b,long long m)
{
D ans;
ans.p=(a.p * b.p % m +a.d * b.d %m *w % m)%m;
ans.d=(a.p * b.d % m +a.d * b.p% m)%m;
return ans;
}
D power(D a,long long b,long long m)
{
D ans;
ans.p = 1;
ans.d = 0;
while (b)
{
if (b & 1)
{
ans=mul(ans,a,m);
}
b/=2;
a=mul(a,a,m);
}
return ans;
}
long long sqre(long long x,long long y)
{
if (y==2) return 1;
if (mypow(x,(y-1)>>1,y)+1 == y)
return -1;
long long a,t;
for (a=1;a<y;a++)
{
t= a * a - x;
w= (t + y) % y;
if (mypow(w,(y-1)>>1,y)+1 == y) break;
}
D tmp;
tmp.p=a;
tmp.d=1;
D ans = power(tmp,(y+1)>>1,y);
return ans.p;
}
int main()
{
sqrt5=sqre(5,mod);
printf("%I64d\n",sqrt5);
long long x,y;
egcd(5,mod,x,y);
x=(x+mod)%mod;
s=(sqrt5*x)%mod;
printf("%I64d\n",s); egcd(2,mod,x,y);
x=(x+mod)%mod;
r1=((sqrt5+1)*x)%mod;
r2=((-sqrt5+1+mod)*x)%mod; printf("%I64d\n",r1);
printf("%I64d\n",r2);
int T;
return 0;
}
系数带入后本题的代码
#include <cstdio>
#include <cstring>
#include <algorithm>
long long s=723398404,r1=308495997,r2=691504013;
long long mod=1000000009;
long long c[100005];
typedef long long Ma[2][2];
void egcd(long long a,long long b,long long &x,long long &y)
{
if (b==0)
{
x=1;
y=0;
return;
}
egcd(b,a%b,x,y);
int t=x;
x=y,y=t-a/b*y;
return;
}
long long mypow(long long x,long long y)
{
long long res=1;
while (y)
{
if (y%2)
res=res * x % mod;
x=x * x % mod;
y/=2;
}
return res;
}
long long acce(long long x,long long y)//等比数列高速求和
{
long long ans=0;
long long powe=x;
long long sum=x;
long long mul=1;
while (y)
{
if (y&1)
{
ans+=mul*sum;
ans%=mod;
mul*=powe;
mul%=mod;
}
sum*=(powe+1);
sum%=mod;
powe*=powe;
powe%=mod;
y/=2;
}
return ans;
}
int main()
{
int T;
long long n,m;
scanf("%d",&T);
while (T--)
{
long long ans=0;
scanf("%lld%lld",&n,&m);
c[0]=1;
long long x,y;
for (long long i=1;i<=m;i++)
{
egcd(i,mod,x,y);
x=(x+mod)%mod;
c[i]=(c[i-1]*x)%mod;
c[i]=(c[i]*(m-i+1))%mod;
}
for (long long i=0;i<=m;i++)
{
x=c[i];
x=x * acce(mypow(r1,i)*mypow(r2,m-i)%mod,n)%mod;
x=(x+mod)%mod;
if ((m-i)%2)
ans=(ans - x + mod)%mod;
else
ans=(ans + x + mod)%mod;
}
ans=ans*mypow(s,m)%mod;
printf("%lld\n",ans);
}
system("pause");
return 0;
}
[zoj 3774]Power of Fibonacci 数论(二次剩余 拓展欧几里得 等比数列求和)的更多相关文章
- 51 Nod 1256 乘法逆元(数论:拓展欧几里得)
1256 乘法逆元 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 收藏 关注 给出2个数M和N(M < N),且M与N互质,找出一个数K满足0 < K ...
- ZOJ Problem Set - 3593 拓展欧几里得 数学
ZOJ Problem Set - 3593 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3593 One Person ...
- ACM数论-欧几里得与拓展欧几里得
ACM数论——欧几里得与拓展欧几里得 欧几里得算法: 欧几里德算法又称辗转相除法,用于计算两个整数a,b的最大公约数. 基本算法:设a=qb+r,其中a,b,q,r都是整数,则gcd(a,b)=gcd ...
- BZOJ-2242 计算器 快速幂+拓展欧几里得+BSGS(数论三合一)
污污污污 2242: [SDOI2011]计算器 Time Limit: 10 Sec Memory Limit: 512 MB Submit: 2312 Solved: 917 [Submit][S ...
- SGU 141.Jumping Joe 数论,拓展欧几里得,二元不等式 难度:3
141. Jumping Joe time limit per test: 0.25 sec. memory limit per test: 4096 KB Joe is a frog who lik ...
- poj 1845 【数论:逆元,二分(乘法),拓展欧几里得,费马小定理】
POJ 1845 题意不说了,网上一大堆.此题做了一天,必须要整理一下了. 刚开始用费马小定理做,WA.(poj敢说我代码WA???)(以下代码其实都不严谨,按照数据要求A是可以等于0的,那么结果自然 ...
- NOIP2012拓展欧几里得
拉板题,,,不说话 我之前是不是说过数据结构很烦,,,我想收回,,,今天开始的数论还要恶心,一早上听得头都晕了 先来一发欧几里得拓展裸 #include <cstdio> void gcd ...
- poj 1061 青蛙的约会+拓展欧几里得+题解
青蛙的约会+拓展欧几里得+题解 纵有疾风起 题意 两只青蛙在网上相识了,它们聊得很开心,于是觉得很有必要见一面.它们很高兴地发现它们住在同一条纬度线上,于是它们约定各自朝西跳,直到碰面为止.可是它们出 ...
- poj 1061 青蛙的约会 拓展欧几里得模板
// poj 1061 青蛙的约会 拓展欧几里得模板 // 注意进行exgcd时,保证a,b是正数,最后的答案如果是负数,要加上一个膜 #include <cstdio> #include ...
随机推荐
- MEMS市场介绍
惠普第一.德州仪器第二 市场观察发展报告说,MEMS市场在2007年增长百分之九,达到70亿美元,其中前30名制造商的收入总和有56亿美元,平均增长7个百分点. 惠普(HP)打印机使用MEMS喷墨头, ...
- Codeforces #344 Div.2
Codeforces #344 Div.2 Interview 题目描述:求两个序列的子序列或操作的和的最大值 solution 签到题 时间复杂度:\(O(n^2)\) Print Check 题目 ...
- iscroll5 版本下的 上拉,下拉 加载数据
上拉时候只是加载第一页的内容,可根据实际情况修改其中的代码. <section id="downwraper" class="nodeBottom bot0 bgf ...
- 充分利用CPU高速缓存,提高程序效率(原理篇)
提高程序效率应该充分利用CPU的高速缓存.要想编写出对CPU缓存友好的程序就得先明白CPU高速缓存的运行机制. i5-2400S: 1.有三级缓存分别为 32k(数据.指令缓存分开,分为32k),25 ...
- VB读写Excel
近期用excel和VB比較多,就简单的学习了一下VB中对Excel的处理.今天就介绍一些吧. 在VB中要想调用Excel,须要打开VB编程环境“project”菜单中的“引用”项目 ...
- 用HtmlLink来改变网站的主题
#region Theme // 注册样式(将主题样式至于通用样式后面) HtmlLink themeCss = new HtmlLink(); themeCss.Href = GetThemeUrl ...
- 第003篇 深入体验C#项目开发(二)
下半本的5个项目也看完了,还是跳着看,只看大概! 第6章 企业交互系统 作者入职一年,开始带新的2个实习生的项目!一个外资企业内部的OA交互系统,这次又是一个基于w ...
- select Into用法
1. insert into 表(列1,列2) select 列1,列2 from 表2 where.... 2.select * into 表2 from 表 1where... 上面写法标识将表1 ...
- Free Sql Server SMSS format Plugin
免费Sql Server 格式化插件 http://www.apexsql.com/sql_tools_refactor.aspx http://architectshack.com/PoorMans ...
- .Net 发邮件
对于.NET而言,从2.0开始,发邮件已经是一件非常easy 的事了.下面我给出一个用C#群发邮件的实例,做了比较详细的注解,希望对有需要的朋友有所help. // 引入命名空间using Syste ...