Java基础查找算法(4)——斐波那契查找

Java基础查找算法(4)——斐波那契查找

代码参考以下B站尚硅谷官方教学视频
尚硅谷-韩顺平图解Java数据结构和算法

1.斐波那契查找简述

斐波那契查找也称黄金分割法,与二分查找和插值查找类似,只是mid的选取有所不同

Java基础查找算法(2)——二分查找(折半查找)

Java基础查找算法(3)——插值查找(插入查找)

斐波那契查找中mid的选取:

黄金分割点:一条线分为两部分,使其中一部分与全长之比等于另一部分与这部分之比。
黄金分割点

黄金分割值 约等于 0.618(取前三位)

斐波那契数列:{1,1,2,3,5,8,13,21,34…} 满足arr[k]=arr[k-1]+arr[k-2] 当前元素等于前两个元素之和

由arr[k]=arr[k-1]+arr[k-2] 得arr[k]-1 = arr[k-1]-1 + arr[k-2]-1 + 1
斐波那契查找mid由来

2.代码实现

斐波那契查找前,需要数组是有序的,调用了之前学过的基数排序代码。
Java基础排序算法(7)——基数排序

package Algorithm.Search;

import Algorithm.Sort.RadixSort;
import Algorithm.Sort.Template;
import Utils.ArrUtil;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

/**
 * 斐波那契查找(黄金分割法)
 */
public class FibonacciSearch {
    private static List<Integer> result = new ArrayList<>();//目标值得下标集合
    static int maxsize = 20;


    /**
     * 非递归初始化该数列
     * 
     * F[n]=F[n-1]+F[n-2]
     */
    public static int[] InitFibonnacci() {
        int[] f = new int[maxsize];
        f[0] = 1;
        f[1] = 1;
        for (int i = 2; i < maxsize; i++) {
            f[i] = f[i - 1] + f[i - 2];
        }
        return f;
    }

    /**
     * 非递归实现斐波那契查找
     * mid=left+F[k-1]-1
     *
     * @param arr    需查找得数组
     * @param target 目标查找值
     * @return 返回目标值下标集合
     */
    public static int search(int[] arr, int target) {
        int left = 0;
        int right = arr.length - 1;
        int k = 0;//斐波那契分割数值的下标 ;K值
        int mid = 0;//存放mid值
        int f[] = InitFibonnacci();//获取斐波那契数列
        while (right > f[k] - 1) k++;//获取斐波那契数列下标
        int[] temp = Arrays.copyOf(arr, f[k]);// //f[k]可能大于a的长度,使用Array类,构造一个新的数组,指向arr[]//不足的部分使用arr的最后一个元素填充
        for (int i = right + 1; i < temp.length; i++) temp[i] = arr[right];
        while (left <= right) {
            mid = left + f[k - 1] - 1;
            if (target < temp[mid]) {//目标值只可能在左侧
                right = mid - 1;
                k--;
            } else if (target > temp[mid]) {//目标值可能在右侧
                left = mid + 1;
                k -= 2;
            } else {//相等
                if (mid < right) return mid;
                else return right;
            }
        }
        return -1;//没有目标值
    }

    public static void main(String[] args) {
        int testSize = 20;
        int[] arr = Template.getIntData(testSize);
        //输出原数组
        ArrUtil.show(arr);
        RadixSort.sort(arr);//需要目标数组有序 调用基数排序
        ArrUtil.show(arr);
        int a = (int) (Math.random() * testSize);
        System.out.println("斐波那契查找" + a);
        int index = search(arr, a);
        System.out.println("下标" + index);

    }
}

3.运行效果

16 11 19 12 0 0 15 19 0 16 15 17 7 19 19 14 10 8 6 11 
最大数19
因此最大位数2
按照第1位,本轮排序结果:0 0 0 10 11 11 12 14 15 15 16 16 6 17 7 8 19 19 19 19 
按照第2位,本轮排序结果:0 0 0 6 7 8 10 11 11 12 14 15 15 16 16 17 19 19 19 19 
0 0 0 6 7 8 10 11 11 12 14 15 15 16 16 17 19 19 19 19 
斐波那契查找8
下标5

代码参考以下B站尚硅谷官方教学视频
尚硅谷-韩顺平图解Java数据结构和算法

发布了67 篇原创文章 · 获赞 32 · 访问量 2万+

猜你喜欢

转载自blog.csdn.net/qq_42391904/article/details/101625665