zoukankan      html  css  js  c++  java
  • 【HDU3117】Fibonacci Numbers

    【HDU3117】Fibonacci Numbers

    题面

    求斐波那契数列的第(n)项的前四位及后四位。

    其中(0leq n<2^{32})

    题解

    前置知识:线性常系数齐次递推

    其实后四位还是比较好求,矩阵快速幂就可以了,主要是前四位。

    先用线性常系数齐次递推求出斐波那契数列的通项公式

    [f_n=frac{sqrt 5}{5}left((frac{1+sqrt5}{2})^n-(frac{1-sqrt5}{2})^n ight) ]

    因为数列的前(39)项我们还是存的下的,所以我们只考虑(ngeq40)的情况

    考虑到(ngeq40)(frac{sqrt 5}{5}*(frac{1-sqrt5}{2})^n)是个很小的东西,可以不考虑它的影响

    那么我们就是要求

    [frac{sqrt 5}{5}(frac{1+sqrt5}{2})^n ]

    现在先考虑这样一个式子,数(x)用科学计数法表示

    [x=t*10^k ]

    那么(x)的前四位即为(t)的前四位,我们将(x)取个常用对数

    [lg x=lg t+k ]

    类比上式以及我们要求的式子:

    [egin{aligned} y&=lgleft(frac{sqrt 5}{5}(frac{1+sqrt5}{2})^n ight)\ &=lgfrac{sqrt 5}{5}+lg;(frac{1+sqrt5}{2})^n\ &=lgfrac{sqrt 5}{5}+n imes lgfrac{1+sqrt5}{2} end{aligned} ]

    那么(lg t=y-lfloor y floor),最后(1000 imes 10^y)的整数部分就是答案。

    代码

    #include <iostream>
    #include <cstdio>
    #include <cstdlib>
    #include <cstring> 
    #include <cmath> 
    #include <algorithm>
    using namespace std; 
    const int Mod = 1e4; 
    struct Matrix { 
    	int m[2][2]; 
    	Matrix() { memset(m, 0, sizeof(m)); } 
    	void init() { for (int i = 0; i < 2; i++) m[i][i] = 1; } 
    	int *operator [] (int id) { return m[id]; } 
    	Matrix operator * (const Matrix &b) { 
    		Matrix res; 
    		for (int i = 0; i < 2; i++) 
    			for (int j = 0; j < 2; j++) 
    				for (int k = 0; k < 2; k++) 
    					res[i][j] = (res[i][j] + m[i][k] * b.m[k][j] % Mod) % Mod; 
    		return res; 
    	} 
    } ; 
    int N, f[40];
    int TASK1() { 
    	double s = log10(sqrt(5.0) / 5.0) + 1.0 * N * log10((1.0 + sqrt(5.0)) / 2.0); 
    	s = s - (int)s; 
    	double ans = 1000 * pow(10.0, s); 
    	return ans; 
    } 
    int TASK2() { 
    	Matrix S, T, ans; int n = N; ans.init(); 
    	S[0][0] = 1; 
    	T[0][0] = 1, T[0][1] = 1; 
    	T[1][0] = 1, T[1][1] = 0; 
    	while (n) { if (n & 1) ans = ans * T; n >>= 1; T = T * T; } 
    	S = ans * S; 
    	return ans[1][0]; 
    } 
    int main () { 
    	f[0] = 0, f[1] = 1; for (int i = 2; i < 40; i++) f[i] = f[i - 1] + f[i - 2]; 
    	while (~scanf("%d", &N)) { 
    		if (N < 40) { printf("%d
    ", f[N]); continue; } 
    		printf("%04d...%04d
    ", TASK1(), TASK2()); 
    	} 
    	return 0; 
    } 
    
  • 相关阅读:
    今天碰到的angular 中的一个小坑
    mvc 防止客服端多次提交
    自定义通用Distinct去除重复数据的2中方式
    Sql 字符串操作类COALESCE
    SQL Server 性能优化
    Visual Studio Tip: Get Public Key Token for a Strong Named Assembly
    C#发送邮件
    Web打印组件jatoolsPrinter(转载)
    SQL SERVER 2005 同步复制技术(转)
    [Asp.net]常见word,excel,ppt,pdf在线预览方案(转)
  • 原文地址:https://www.cnblogs.com/heyujun/p/10298953.html
Copyright © 2011-2022 走看看