Fibonacci(N)=Fibonacii(N-1)+Fibonacci(N-2)

其中 Fibonacci(0)=0;Fibonacci(1)=1

用循环或则递归实现Fibonacci算法很简单,这里就不说了,如果要用公式实现的话,需要进行开根号和幂运算,普通的long型号只能精确到小数点之后的16位,这就意味着当N很大的时候由于小数位overflow无法进行精确运算,本人测试当N大于92时候因为数据溢出的原因就与实际结果不一样了,如果小数点后面不是精确16位,而是自己设定的话,那么运算结果会精确到自己的期望值。JAVA为我们提供了BigDecimal这个类,BigDecimal类提供了幂运算,但是没有提供开平方运算,Fibonacci公式中的开平方运算要自己实现。这里主要介绍一下自己是如何实现开平方运算的

private static BigDecimal mysqrt(BigDecimal a, int scale) {

        BigDecimal x = new BigDecimal(Math.sqrt(a.doubleValue()),
MathContext.DECIMAL64);
if (scale < 17)
return x; BigDecimal b2 = new BigDecimal("2");
for (int tempScale = 16; tempScale < scale; tempScale *= 2) {
// x = x - (x * x - a) / (2 * x);
x = x.subtract(x.multiply(x).subtract(a)
.divide(x.multiply(b2), scale, BigDecimal.ROUND_HALF_EVEN));
}
return x; }

这里用到了牛顿线性迭代算法,如果不明白的话可以在百度搜索。返回的结果是小数点后精确到scale位的数字。

现在把所有代码给出

package com.fujitsu.soft.rad.devsemi.fibonacci3;

import java.math.BigDecimal;
import java.math.MathContext; public class Fibonacci { /*
* by l.zhang
*/
public static void main(String[] args) { System.out.println(fibonacci(100)); } /*
* by l.zhang
*/ public static String fibonacci(int i) {// 大きい桁表現すればlong 2動作の問題 if (i < 0 || i > 1000) {
System.out.println("please input a new number ");
return "-1";
} else if (i == 0) { return "0";
} else if (i == 1) {
return "1";
} else {
return siki(i); } } /*
* 公式で計算 i>=2 by l.zhang
*/
private static String siki(int i) { BigDecimal temp1 = mysqrt(new BigDecimal("5.0"), 100);// Math.pow(5.0,
// 0.5)
BigDecimal temp2 = temp1.divide(new BigDecimal("5.0"));// Math.pow(5.0,
// 0.5) / 5.0;
BigDecimal temp3 = new BigDecimal("1.0");
BigDecimal temp4 = temp3.add(temp1);// (1 + Math.pow(5.0, 0.5))
BigDecimal temp5 = temp3.subtract(temp1);// (1 - Math.pow(5.0, 0.5))
BigDecimal temp6 = temp4.divide(new BigDecimal("2.0"));// (1 +
// Math.pow(5.0,
// 0.5)) / 2.0
BigDecimal temp7 = temp5.divide(new BigDecimal("2.0"));// (1 -
// Math.pow(5.0,
// 0.5)) / 2.0
BigDecimal temp8 = temp6.pow(i, new MathContext(100));// Mypow(temp6,
// i);//
// Math.pow((1 +
// Math.pow(5.0, 0.5))
// / 2.0, i) BigDecimal temp9 = temp7.pow(i, new MathContext(100));// Mypow(temp7,
// i);//
// Math.pow((1 -
// Math.pow(5.0, 0.5))
// / 2.0, i);
// double temp2 = Math.pow((1 + Math.pow(5.0, 0.5)) / 2.0, i);
// double temp3 = Math.pow((1 - Math.pow(5.0, 0.5)) / 2.0, i);
BigDecimal temp10 = temp8.subtract(temp9);
BigDecimal result = temp2.multiply(temp10); String str_res = result.toString();
System.out.println(str_res);
int index = str_res.indexOf(".", 0); return str_res.substring(0, index);
// (temp1 * (temp2 - temp3));
} private static BigDecimal mysqrt(BigDecimal a, int scale) { BigDecimal x = new BigDecimal(Math.sqrt(a.doubleValue()),
MathContext.DECIMAL64);
if (scale < 17)
return x; BigDecimal b2 = new BigDecimal("2");
for (int tempScale = 16; tempScale < scale; tempScale *= 2) {
// x = x - (x * x - a) / (2 * x);
x = x.subtract(x.multiply(x).subtract(a)
.divide(x.multiply(b2), scale, BigDecimal.ROUND_HALF_EVEN));
}
return x; } }

用BigDecimal类实现Fibonacci算法的更多相关文章

  1. 算法笔记--java的BigInteger类及BigDecimal类

    引包:import java.math.*; BigInteger类: 可以使用构造方法:public BigInteger(String val),或者valueOf(int)函数,如: BigIn ...

  2. java中小数的处理:高精度运算用bigDecimal类,精度保留方法,即舍入方式的指定

    一. 计算机的小数计算一定范围内精确,超过范围只能取近似值: 计算机存储的浮点数受存储bit位数影响,只能保证一定范围内精准,超过bit范围的只能取近似值. java中各类型的精度范围参见:http: ...

  3. 用C# BigInteger实现的BigDecimal类,终于可以直接做四则运算了。

    https://code.google.com/p/dotnet-big-decimal/ 这是个BigDecimal类的开源项目,支持Operators +, - and *. 俺给改了改,加上了除 ...

  4. java基础-BigDecimal类常用方法介绍

    java基础-BigDecimal类常用方法介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.BigDecimal类概述 我们知道浮点数的计算结果是未知的.原因是计算机二进制 ...

  5. JAVA集合类简要笔记 - 内部类 包装类 Object类 String类 BigDecimal类 system类

    常用类 内部类 成员内部类.静态内部类.局部内部类.匿名内部类 概念:在一个类的内部再定义一个完整的类 特点: 编译之后可生成独立的字节码文件 内部类可直接访问外部类私有成员,而不破坏封装 可为外部类 ...

  6. Java:利用BigDecimal类巧妙处理Double类型精度丢失

    目录 本篇要点 经典问题:浮点数精度丢失 十进制整数如何转化为二进制整数? 十进制小数如何转化为二进制数? 如何用BigDecimal解决double精度问题? new BigDecimal(doub ...

  7. BigDecimal类

    如果需要精确的计算结果,则必须使用BigDecimal类,而且使用BigDecimal类也可以进行大数的操作. //========================================== ...

  8. Java大数处理类:BigInteger类和BigDecimal类

    当我们要处理非常大的数据时,平常用的数据类型已不足以表示,在Java中有两个类BigInteger和BigDecimal分别表示大整数类和大浮点数类,这两个类在理论上只要计算机内存足够大就能够表示无线 ...

  9. BIgInteger类和BigDecimal类的理解

    第一部分: 这两个类位于java.math包内,要使用它们必须在类前面引用该包:import java.math.BigInteger;和import java.math.BigDecimal; Bi ...

随机推荐

  1. VB6-系统打印常识

    在一次做图片打印的时候,对位置的调整老是不得法,后来通过CBM666老师的帮助才解决问题,分享以下他给的帮助.     , , picA.Width , picA.Height Printer.End ...

  2. Starting MySQL.. ERROR! The server quit without updating PID file (/usr/local/mysql/data/localhost.localdomain.pid).

      [root@localhost ~]# cd /usr/local/mysql   [root@localhost mysql]# chown -R mysql.mysql . [root@loc ...

  3. IOS中将十进制色值转换成UIColor

    最近因项目需要,在网上找了一些代码,整合了一下,实现的效果就是将10进制的RGB色值转换IOS用的UIColor,方法还有缺陷,有待改进 UIColor *getColorFromString(NSS ...

  4. 【BZOJ】1015: [JSOI2008]星球大战starwar

    1015: [JSOI2008]星球大战starwar 题意:一个点数为N(1<= 40w),边数为M(1<=20w)的图,总共删除k个节点,问开始以及每次删除一个节点之后图的连通块数? ...

  5. linux之vim编辑器

    Vi简介1. Vi是一种广泛存在于各种UNIX和Linux系统中的文本编辑程序.2. Vi不是排版程序,只是一个纯粹的文本编辑程序.3. Vi是全屏幕文本编辑器,它没有菜单,只有命令.4. Vi不是基 ...

  6. MVC-内容详情页显示内容

    @model InfoDataProvider.DataModel.FAQ_ContentUser 内容Content字段:如果里面有html标签. @Html.DisplayFor(p => ...

  7. UIMenuController搭配UIPasteboard,执行拷贝-黏贴操作-b

    一.基本概念 UIKit框架中,可以直接执行拷贝黏贴操作的有:UITextView.UITextField和UIWebView,其他控件需要实现相关方法. 关于UIPasteboard ·黏贴板是ap ...

  8. java.sql.SQLException: Io 异常: Connection refused(DESCRIPTION=(TMP=)(VSNNUM=186646784)(ERR=12505)(ERR

    dbc 链接orcal出错 java.sql.SQLException: Io 异常: Connection refused(DESCRIPTION=(TMP=)(VSNNUM=186646784)( ...

  9. [转载]如何打一手好Log

    如果项目上过线的话,那你一定知道Log是多么重要. 为什么说Log重要呢?因为上线项目不允许你调试,你只能通过Log来分析问题.这时打一手好Log的重要性绝不亚于写一手好代码.项目出问题时,你要能拿出 ...

  10. 1012: [JSOI2008]最大数maxnumber

    单点更新,区间求最大值的题: 可以使用树状数组和线段树: #include<cstdio> #include<cstring> #include<algorithm> ...