斐波那契查找

斐波那契查找

      • 概述
      • 步骤
      • 代码示例
      • 输出结果

概述

斐波那契查找是一种基于斐波那契数列的查找算法,用于在有序数组中查找目标元素的位置。与二分查找类似,斐波那契查找也是一种分治算法,它通过比较目标值与数组的中间元素来确定下一步的查找范围。

在介绍斐波那契查找算法之前,我们先介绍一下很它紧密相连并且大家都熟知的一个概念——黄金分割。

黄金比例又称黄金分割,是指事物各部分间一定的数学比例关系,即将整体一分为二,较大部分与较小部分之比等于整体与较大部分之比,其比值约为1:0.618或1.618:1。

0.618被公认为最具有审美意义的比例数字,这个数值的作用不仅仅体现在诸如绘画、雕塑、音乐、建筑等艺术领域,而且在管理、工程设计等方面也有着不可忽视的作用。因此被称为黄金分割。

在数学中有一个非常有名的数学规律:斐波那契数列:1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89…….

(从第三个数开始,后边每一个数都是前两个数的和)。

然后我们会发现,随着斐波那契数列的递增,前后两个数的比值会越来越接近0.618,利用这个特性,我们就可以将黄金比例运用到查找技术中。

img

基本思想:也是二分查找的一种提升算法,通过运用黄金比例的概念在数列中选择查找点进行查找,提高查找效率。同样地,斐波那契查找也属于一种有序查找算法。

斐波那契查找也是在二分查找的基础上进行了优化,优化中间点mid的计算方式即可

步骤

算法步骤如下:

  1. 首先,确定一个斐波那契数列。斐波那契数列的前两个数是1和1,后续的数是前两个数之和(1, 1, 2, 3, 5, 8, 13, 21, …)。

  2. 找到一个等于或大于数组长度的斐波那契数F[k],其中k是一个非负整数。可以通过递推或迭代计算斐波那契数列,直到找到满足条件的F[k]。

  3. 将数组扩展为长度为F[k],并将多出的位置值设为数组最后一个元素的值(即将数组填充到斐波那契数列的长度)。

  4. 设置两个指针:低指针low和高指针high,初始时分别指向数组的首位和末位。

  5. 比较目标值与数组的中间元素arr[mid],其中mid = low + F[k-1] - 1。

    • 如果目标值等于arr[mid],则找到了目标元素,返回mid。

    • 如果目标值小于arr[mid],则目标元素在低指针low和mid之间,将高指针high更新为mid-1,并将k减1。

    • 如果目标值大于arr[mid],则目标元素在mid和高指针high之间,将低指针low更新为mid+1,并将k减2。

  6. 重复步骤5,直到找到目标元素或低指针low大于高指针high为止。

  7. 如果找到目标元素,返回其位置;否则,返回-1,表示目标元素不存在于数组中。

斐波那契查找的时间复杂度为O(log n),与二分查找相当,但在某些情况下,斐波那契查找具有更好的性能表现。它适用于较大的有序数组,并且对于元素分布不均匀的情况下,查找性能更稳定。

代码示例

需求:定义一个方法利用斐波那契查找,数据如下:{2, 5, 8, 12, 16, 23, 38, 56, 72, 91};

要求:查询某个元素是否存在

代码如下:

package text.text02;

/*
斐波那契查找:
    也是二分查找的一种提升算法,通过运用黄金比例的概念在数列中选择查找点进行查找,提高查找效率。同样地,斐波那契查找也属于一种有序查找算法。

斐波那契查找也是在二分查找的基础上进行了优化,优化中间点mid的计算方式即可


 */
public class text09A {
    public static void main(String[] args) {
        int[] arr = {2, 5, 8, 12, 16, 23, 38, 56, 72, 91};

        //定义两个要查询的数(一个能查到,一个查不到)
        int target1 = 23;
        int target2 = 48;
        //调用fibonacciSearch方法
        int result1 = fibonacciSearch(arr, target1);
        int result2 = fibonacciSearch(arr, target2);
        //调用judge方法,并将fibonacciSearch方法的返回值和要查找的数作为参数
        judge(result1, target1);       //23存在数组中,其索引为:5
        judge(result2, target2);       //48不存在于数组中

    }

    public static int fibonacciSearch(int[] arr, int target) {
        //定义一个变量记录数组的总长度
        int n = arr.length;

        //初始化斐波那契数列
        int fib2 = 0; // 第二个斐波那契数初始化为0
        int fib1 = 1; // 第一个斐波那契数初始化为1
        int fib = fib1 + fib2; // 当前斐波那契数

        //构建斐波那契数列,使得最大斐波那契数不超过数组长度
        //循环直到找到最大的斐波那契数
        while (fib < n) {
            fib2 = fib1; //将第二个斐波那契数更新为前一个斐波那契数
            fib1 = fib;  //将第一个斐波那契数更新为当前斐波那契数
            fib = fib1 + fib2;   // 计算新的当前斐波那契数
        }

        int offset = -1; // 保存数组的偏移量
        //斐波那契数列中进行迭代搜索
        while (fib > 1) {
            int i = Math.min(offset + fib2, n - 1);  //计算当前位置

            //根据当前位置与目标值的关系进行调整。
            //如果找到目标值,返回索引位置
            if (arr[i] == target) {
                return i;
            }
            //如果当前值小于目标值,调整斐波那契数列以向后搜索
            else if (arr[i] < target) {
                fib = fib1;
                fib1 = fib2;
                fib2 = fib - fib1;
                offset = i;
            }
            //如果当前值大于目标值,调整斐波那契数列以向前搜索
            else {
                fib = fib2;
                fib1 = fib1 - fib2;
                fib2 = fib - fib1;
            }
        }
        //检查当前斐波那契数是否为1,并且下一个元素是否为目标值。
        //如果是,返回偏移量加1的索引位置。
        if (fib1 == 1 && arr[offset + 1] == target) {
            return offset + 1;
        }
        //如果未找到目标值,返回-1表示未找到。
        return -1;
    }

    //定义一个方法根据interpolationSearch方法的返回值判断
    public static void judge(int index, int number) {
        if (index == -1) {
            System.out.println(number + "不存在于数组中");
        } else {
            System.out.println(number + "存在数组中,其索引为:" + index);
        }
    }
}

在这个示例中,我们定义了一个名为fibonacciSearch的静态方法,接受一个有序整数数组arr和要查找的目标值target作为参数。

在方法内部,我们首先确定一个不小于数组长度的斐波那契数fib,并且获取斐波那契数列中的第二个数fib2和第一个数fib1

接下来,我们将当前的斐波那契数fib作为查找区间的大小。通过迭代,我们不断更新fib1fib2fib,直到fib不小于数组长度。此时,fib2表示最后一个小于或等于数组长度的斐波那契数。

然后,我们使用偏移量offset来跟踪查找区间的起始位置,在每一次迭代中,我们根据目标值与数组中间元素的比较结果,将查找区间调整到适当的位置。

如果找到目标值,我们返回其位置;如果斐波那契数fib1为1并且偏移位置的下一位等于目标值,我们也返回该位置;否则,返回-1,表示目标值不存在于数组中。

main方法中,我们定义了一个示例数组arr和要查找的目标值target,然后调用fibonacciSearch方法来执行查找操作。如果找到目标值,则打印其索引;否则,打印目标值不存在于数组中的消息。

输出结果

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/340452.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++版QT:鼠标事件

鼠标常用的事件可以说有一下几种&#xff1a;鼠标按下、鼠标移动、鼠标移动、鼠标双击和鼠标滚轮事件。 当你想使用他们&#xff0c;需要包含头文件&#xff1a;#include <QMouseEvent> 需要对鼠标事件进行处理时&#xff0c;通常要重新实现以下几个鼠标事件处理函数&a…

设备对象(DEVICE_OBJECT)

设备对象(DEVICE_OBJECT) 每个驱动程序会创建一个或多个设备对象&#xff0c;用DEVICE_OBJECT数据结构表示。每个设备对象都会有一个指针指向下一个设备对象&#xff0c;因此就形成一个设备链。设备对象链的第一个设备是由DRIVER_OBJECT结构体中指明的。设备对象保存设…

UI自动化中的option选项配置

&#x1f525; 交流讨论&#xff1a;欢迎加入我们一起学习&#xff01; &#x1f525; 资源分享&#xff1a;耗时200小时精选的「软件测试」资料包 &#x1f525; 教程推荐&#xff1a;火遍全网的《软件测试》教程 &#x1f4e2;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1…

《WebKit 技术内幕》学习之七(2): 渲染基础

2 网页层次和RenderLayer树 2.1 层次和RenderLayer对象 前面章节介绍了网页的层次结构&#xff0c;也就是说网页是可以分层的&#xff0c;这有两点原因&#xff0c;一是为了方便网页开发者开发网页并设置网页的层次&#xff0c;二是为了WebKit处理上的便利&#xff0c;也就是…

C++中命名空间、缺省参数、函数重载

目录 1.命名空间 2.缺省参数 3.函数重载 1.命名空间 在C中定义命名空间我们需要用到namespace关键字&#xff0c;后面跟上命名空间的名字&#xff0c;结构框架有点类似结构体&#xff08;如图所示&#xff09; 上面的代码我一一进行讲解&#xff1a; 1.我们先来说第三行和main函…

开源堡垒机JumpServer本地安装并配置公网访问地址

文章目录 前言1. 安装Jump server2. 本地访问jump server3. 安装 cpolar内网穿透软件4. 配置Jump server公网访问地址5. 公网远程访问Jump server6. 固定Jump server公网地址 前言 JumpServer 是广受欢迎的开源堡垒机&#xff0c;是符合 4A 规范的专业运维安全审计系统。JumpS…

基于CLIP4Clip的DRL的WTI模块实现

关于DRL的WTI模块&#xff1a; Weighted Token-wise Interaction&#xff1a; 直觉上&#xff0c;并非所有的单词和视频帧都同等重要。我们提供一种自适应方法&#xff0c;来调整每个标记的权重大小&#xff1a; 注&#xff1a;其中两个f函数都是MLP和softmax构成。 WTI的算…

使用STM32的SPI接口实现与外部传感器的数据交互

一、引言 外部传感器是嵌入式系统中常用的外设&#xff0c;用于检测环境参数、采集数据等。通过STM32微控制器的SPI接口&#xff0c;可以与外部传感器进行数据交互&#xff0c;从而实现数据的采集和控制。本文将介绍如何使用STM32的SPI接口实现与外部传感器的数据交互&#xff…

云计算任务调度仿真05

今天再分享一个新的调度框架deeprm 本项目基于hongzimao/deeprm&#xff0c;原作者还著有论文Resource Management with Deep Reinforcement Learning 。 这个框架研究的也蛮多的&#xff0c;我在一篇博士论文中也看到了基于此的研究工作&#xff0c;但是论文题目忘记了。 运…

【C++】入门(一)

前言&#xff1a; 本篇博客将带大家认识C&#xff0c;熟悉基本语法 文章目录 认识CC的诞生与发展C 在行业中的运用 一、命名空间1.1 命名空间的定义1.2 命名空间的使用1.3 命名空间的访问 二、C输入&输出输出操作符 <<输入操作符 >>换行符和刷新输出缓冲区关键…

如何在CentOS8使用宝塔面板本地部署Typecho个人网站并实现公网访问【内网穿透】

文章目录 前言1. 安装环境2. 下载Typecho3. 创建站点4. 访问Typecho5. 安装cpolar6. 远程访问Typecho7. 固定远程访问地址8. 配置typecho 前言 Typecho是由type和echo两个词合成的&#xff0c;来自于开发团队的头脑风暴。Typecho基于PHP5开发&#xff0c;支持多种数据库&#…

vmware 安装Rocky-9.3系统

安装系统截图 安装完成&#xff0c;启动 查看版本和内核 开启远程登陆授权 1、编辑配置文件 #提升权限&#xff0c;输入su,并输入密码 su #编辑ssh文件开启root远程登陆 vi /etc/ssh/sshd_config找到以下内容&#xff1a;#PermitRootLogin prohibit-password 添加&#xff1a…

C#winform上位机开发学习笔记5-串口助手的定时发送功能添加

1.功能描述 选择自动发送功能后&#xff0c;按照设定的发送时间发送发送框中的信息数据&#xff0c;设定时间可以手动输入&#xff0c;当手动输入信息无效&#xff08;非数字&#xff09;时&#xff0c;系统弹出错误提示&#xff0c;并将其设置为默认定时时间。 2.代码部分 步…

【MySQL进阶】视图_存储过程_存储函数_触发器

文章目录 视图基本介绍视图操作视图创建视图查询视图修改视图删除 存储过程基本介绍基本操作存储语法变量IF语句参数传递CASEWHILEREPEATLOOP游标 存储函数触发器基本介绍基本操作 总结 视图 基本介绍 视图概念&#xff1a;视图是一种虚拟存在的数据表&#xff0c;这个虚拟的表…

【Linux对磁盘进行清理、重建、配置文件系统和挂载,进行系统存储管理调整存储结构】

Linux 调整存储结构 前言一、查看磁盘和分区列表二、创建 ext4 文件系统&#xff0c;即&#xff1a;格式化分区为ext4文件系统。1.使用命令 mkfs.ext4 (make file system)报错如下&#xff1a;解决办法1&#xff1a;&#xff08;经测试&#xff0c;不采用&#xff09;X解决办法…

Docker-Jenkins编译android-app的两种方案

Docker-Jenkins编译android-app的两种方案 android开发使用jenkins编译&#xff0c;自动集成修改点/自动命名/自动备份&#xff0c;将修改的apk发布到测试服务器发布网盘&#xff0c;而不需要用通讯工具传来传去。 jenkins用在互联网开发编译比较常见&#xff0c;如果android开…

电力电子技术

文章目录 5 直流直流变流电路5.0 简介5.1 基本斩波电路5.1.1 降压斩波电路 Buck Chopper5.1.1.1 小纹波近似 5.1.2升压斩波电路 11 DC-DC 变换器数字控制11.1 基于单片机控制11.2 基于 DSP 控制11.3 基于 FPGA 控制 12 多相交错并联拓扑结构12.1 多相交错并联12.1 多相交错并联…

CS8370错误,这是由于使用了C# 7.3中不支持的功能

目录 背景: 第一种方法: 第二种办法: 背景: 在敲代码的时候&#xff0c;程序提示报错消息提示:CS8370错误&#xff0c;那么这是什么原因导致的&#xff0c;这是由于使用了C# 7.3中不支持的功能&#xff0c;不支持该功能&#xff0c;那就是版本太低我们就需要升级更高的版本&…

韩国访问学者如何申请?

作为访问学者&#xff0c;前往韩国学术机构进行研究是一项令人期待的经历。首先&#xff0c;申请者需要查找目标学术机构的官方网站&#xff0c;下面知识人网小编带大家了解其访问学者计划的具体要求和申请流程。 通常&#xff0c;申请程序包括填写在线申请表格&#xff0c;提交…

《WebKit 技术内幕》学习之八(3):硬件加速机制

3 其他硬件加速模块 3.1 2D图形的硬件加速机制 其实网页中有很多绘图操作是针对2D图形的&#xff0c;这些操作包括通常的网页绘制&#xff0c;例如绘制边框、文字、图片、填充等&#xff0c;它们都是典型的2D绘图操作。在HTML5中&#xff0c;规范又引入了2D绘图的画布功能&a…