斐波那契(黄金分割)查找

该博客介绍了一种利用斐波那契数列改进的搜索算法——斐波那契搜索。首先,代码展示了如何生成一个大小为20的斐波那契数列,然后详细解释了如何在非递归方式下应用该数列进行查找操作。通过调整斐波那契分割点,算法能够在数组中高效地定位目标值,减少了比较次数。
摘要由CSDN通过智能技术生成

思路: 

 代码: 

public class FibonacciSearch {
    public static int maxSize = 20;
    public static void main(String[] args) {
        int[] arr = {1,8,10,89,1000,1234,12345};
        System.out.println(fibSearch(arr, 89));
    }
    // 因为后面我们mid = low+F(k - 1)-1. 需要使用斐波那契数列, 因此我们需要先获取到斐波那契数列
    // 用非递归的方法得到一个大小为20的斐波那契数列
    public static int[] fbn(){
        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;
    }

    // 非递归的方式编写斐波那契数列查找算法

    /**
     *
     * @param a
     * @param key 需要查找的值
     * @return 返回对应的下标 没有则返回-1
     */
    public static int fibSearch(int[] a, int key){
        int low = 0;
        int high = a.length -1;
        int k = 0; // 表示斐波那契分割数值的下标  mid = low+F(k - 1)-1
        int mid = 0;
        int[] f = fbn(); // 获取斐波那契数列
        // 获取斐波那契分割数值的下标
        while (high > f[k]-1){
            k++;
        }
        // 因为 f[k] 可能大于数组a 的长度 需要一个Array类 构造一个新的数组 并指向temp[]
        int[] temp = Arrays.copyOf(a, f[k]); // 多出来的部分用0补起来
        // 实际上需要使用a最后的数填充temp
        // 举例 :
        // temp:{1,8,10,89,1000,1234,0,0} =>temp:{1,8,10,89,1000,1234,1234}
        for (int i = high + 1;i < temp.length; i++){
            temp[i] = a[high];
        }

        // 使用while循环查找我们的key
        while (low <= high) {
            mid = low + f[k-1]-1;
            if (key < temp[mid]){//说明应该向数组的前一部分查找(左边)
                high = mid -1;
                // 为什么是k--
                // 1 全部元素 = 前面的元素 + 后边的元素
                // 2 f[k] = f[k-1] + f[k-2];
                // 因为前面有f[k-1]个元素,所以我们继续拆分   f[k-1] = f[k-2] + f[k-3];
                // 即在f[k-1] 的前面继续查找 k--
                // 即下次在循环的时候 mid = f[k-1-1] -1
                k--;
            }else if ( key > temp[mid]){ // 我们继续向数组的右边查找
                low = mid + 1;
                // 为什么是k-2
                // 1 全部元素 = 前面的元素 + 后边的元素
                // 2 f[k] = f[k-1] + f[k-2];
                // 3 因为后面有f[k-2] 所以加足拆分 f[k -2] = f[k-3] + f[k-4];
                // 4 即在f[k-2] 的前面继续查找
                // 下次循环mid = f[k-1-2] -1
                k -= 2;
            }else { // 找到
                // 需要确定返回的是哪个下标
                if (mid <= high){
                    return mid;
                }else {
                    return high;
                }
            }
        }
        return -1; // 没有找到

    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值