详解Java Fibonacci Search斐波那契搜索算法代码实现

 更新时间:2020年10月13日 09:53:24   作者:失控的狗蛋~  
这篇文章主要介绍了详解Java Fibonacci Search斐波那契搜索算法代码实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一, 斐波那契搜索算法简述

斐波那契搜索(Fibonacci search) ,又称斐波那契查找,是区间中单峰函数的搜索技术。

斐波那契搜索采用分而治之的方法,其中我们按照斐波那契数列对元素进行不均等分割。此搜索需要对数组进行排序。

与二进制搜索不同,在二进制搜索中,我们将元素分成相等的两半以减小数组范围-在斐波那契搜索中,我们尝试使用加法或减法来获得较小的范围。

斐波那契数列的公式是:

Fibo(N)=Fibo(N-1)+Fibo(N-2)

此系列的前两个数字是Fibo(0) = 0和Fibo(1) = 1。因此,根据此公式,该级数看起来像是0、1、1、2、3、5、8、13、21。。。这里要注意的有趣观察是:

  • Fibo(N-2) 大约是1/3的 Fibo(N)
  • Fibo(N-1) 大约是2/3的 Fibo(N)

因此,当我们使用斐波那契数列来划分范围时,它会以与上述相同的比率进行分割。

二,斐波那契搜索算法代码实现

/**
  * 
  * @param integers
  * @param elementToSearch
  * @return
  */
 public static int fibonacciSearch(int[] integers, int elementToSearch) {

  int fibonacciMinus2 = 0;
  int fibonacciMinus1 = 1;
  int fibonacciNumber = fibonacciMinus2 + fibonacciMinus1;
  int arrayLength = integers.length;

  while (fibonacciNumber < arrayLength) {
   fibonacciMinus2 = fibonacciMinus1;
   fibonacciMinus1 = fibonacciNumber;
   fibonacciNumber = fibonacciMinus2 + fibonacciMinus1;
  }

  int offset = -1;

  while (fibonacciNumber > 1) {
   int i = Math.min(offset+fibonacciMinus2, arrayLength-1);

   if (integers[i] < elementToSearch) {
    fibonacciNumber = fibonacciMinus1;
    fibonacciMinus1 = fibonacciMinus2;
    fibonacciMinus2 = fibonacciNumber - fibonacciMinus1;
    offset = i;
   }

   else if (integers[i] > elementToSearch) {
    fibonacciNumber = fibonacciMinus2;
    fibonacciMinus1 = fibonacciMinus1 - fibonacciMinus2;
    fibonacciMinus2 = fibonacciNumber - fibonacciMinus1;
   }

   else return i;
  }

  if (fibonacciMinus1 == 1 && integers[offset+1] == elementToSearch)
   return offset+1;

  return -1;
 }

 三,斐波那契搜索算法总结

首先从找到斐波那契数列中最接近但大于数组长度的数字开始。这fibonacciNumber是在13刚好大于数组长度10时发生的。

接下来,我们比较数组的元素,并根据该比较,执行以下操作之一:

  • 将要搜索的元素与处的元素进行比较fibonacciMinus2,如果值匹配,则返回索引。
  • 如果elementToSearch比当前元素时,我们移动在斐波纳契数列上一步,而改变的值fibonacciNumber,fibonacciMinus1与fibonacciMinus2相应。偏移量将重置为当前索引。
  • 如果elementToSearch比当前元素小,我们继续前进后退两步在斐波纳契数列和改变的值fibonacciNumber,fibonacciMinus1与fibonacciMinus2相应。

输出结果:

时间复杂度

此搜索的最坏情况时间复杂度为O(log(N))。

空间复杂度

虽然我们需要将三个数字保存在斐波那契数列中并要搜索的元素,但我们需要四个额外的空间单位。

对空间的要求不会随着输入数组的大小而增加。因此,可以说斐波那契搜索的空间复杂度为O(1)。

当除法运算是CPU要执行操作时,将使用此搜索。二进制搜索之类的算法由于使用除法对数组进行划分,因此效果较差。

这种搜索的另一个好处是当输入数组的元素无法放入RAM中时。在这种情况下,此算法执行的局部操作范围可帮助其更快地运行。

 四,跳转搜索算法完整代码

  If you are interested, try it.

public class SearchAlgorithms {

 /**
  *
  * @param integers
  * @param elementToSearch
  * @return
  */
 public static int fibonacciSearch(int[] integers, int elementToSearch) {

  int fibonacciMinus2 = 0;
  int fibonacciMinus1 = 1;
  int fibonacciNumber = fibonacciMinus2 + fibonacciMinus1;
  int arrayLength = integers.length;

  while (fibonacciNumber < arrayLength) {
   fibonacciMinus2 = fibonacciMinus1;
   fibonacciMinus1 = fibonacciNumber;
   fibonacciNumber = fibonacciMinus2 + fibonacciMinus1;
  }

  int offset = -1;

  while (fibonacciNumber > 1) {
   int i = Math.min(offset+fibonacciMinus2, arrayLength-1);

   if (integers[i] < elementToSearch) {
    fibonacciNumber = fibonacciMinus1;
    fibonacciMinus1 = fibonacciMinus2;
    fibonacciMinus2 = fibonacciNumber - fibonacciMinus1;
    offset = i;
   }

   else if (integers[i] > elementToSearch) {
    fibonacciNumber = fibonacciMinus2;
    fibonacciMinus1 = fibonacciMinus1 - fibonacciMinus2;
    fibonacciMinus2 = fibonacciNumber - fibonacciMinus1;
   }

   else return i;
  }

  if (fibonacciMinus1 == 1 && integers[offset+1] == elementToSearch)
   return offset+1;

  return -1;
 }
 /**
  * 打印方法
  * @param elementToSearch
  * @param index
  */
 public static void print(int elementToSearch, int index) {
  if (index == -1){
   System.out.println(elementToSearch + " 未找到");
  }
  else {
   System.out.println(elementToSearch + " 在索引处找到: " + index);
  }
 }
 //测试一下
 public static void main(String[] args) {
  int index = fibonacciSearch(new int[]{3, 22, 27, 47, 57, 67, 89, 91, 95, 99}, 67);
  print(67, index);
 }
}

到此这篇关于详解Java Fibonacci Search斐波那契搜索算法代码实现的文章就介绍到这了,更多相关Java Fibonacci Search 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • java通过模拟post方式提交表单实现图片上传功能实例

    java通过模拟post方式提交表单实现图片上传功能实例

    这篇文章主要介绍了java通过模拟post方式提交表单实现图片上传功能实例,涉及Java针对表单的提交操作响应及文件传输的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-11-11
  • Spring AI聊天功能开发步骤

    Spring AI聊天功能开发步骤

    本文给大家介绍Spring AI聊天功能开发步骤,首先引入依赖,继承父版本的springboot依赖,最好是比较新的依赖,结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧
    2024-04-04
  • 通过netty把百度地图API获取的地理位置从Android端发送到Java服务器端的操作方法

    通过netty把百度地图API获取的地理位置从Android端发送到Java服务器端的操作方法

    这篇文章主要介绍了通过netty把百度地图API获取的地理位置从Android端发送到Java服务器端,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-10-10
  • java多线程编程制作电子时钟

    java多线程编程制作电子时钟

    本文给大家汇总了几个使用java多线程编程实现的电子时钟的代码,思路非常的巧妙,也都很实用,有需要的小伙伴可以参考下。
    2015-11-11
  • SpringBoot开发实战之自动配置

    SpringBoot开发实战之自动配置

    SpringBoot的核心就是自动配置,自动配置又是基于条件判断来配置Bean,下面这篇文章主要给大家介绍了关于SpringBoot开发实战之自动配置的相关资料,需要的朋友可以参考下
    2021-08-08
  • 解决JD-GUI for mac big sur打不开问题

    解决JD-GUI for mac big sur打不开问题

    这篇文章主要介绍了解决JD-GUI for mac big sur打不开问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-01-01
  • Java多线程中的CyclicBarrier使用方法详解

    Java多线程中的CyclicBarrier使用方法详解

    这篇文章主要介绍了Java多线程中的CyclicBarrier使用方法详解,CyclicBarrier是一种同步辅助工具,它允许一组线程都等待对方到达公共障碍点,在涉及固定大小的线程的程序中,CyclicBarriers非常有用,这些线程间必须相互等待,需要的朋友可以参考下
    2023-12-12
  • Java实现矩形碰撞检测

    Java实现矩形碰撞检测

    这篇文章主要为大家详细介绍了Java实现矩形碰撞检测,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-06-06
  • 如何将默认的maven仓库改为阿里的maven仓库

    如何将默认的maven仓库改为阿里的maven仓库

    这篇文章主要介绍了如何将默认的maven仓库改为阿里的maven仓库,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-12-12
  • Java中LinkedHashSet、LinkedHashMap源码详解

    Java中LinkedHashSet、LinkedHashMap源码详解

    这篇文章主要介绍了Java中LinkedHashSet、LinkedHashMap源码详解,LinkedHashMap是一个以双向链表的方式将Entry节点链接起来的HashMap子类,它在HashMap的基础上实现了更多的功能,具有顺序存储和遍历的特性,需要的朋友可以参考下
    2023-09-09

最新评论