四大查找算法

查找算法

1.线性查找算法

介绍:线性查找也成为顺序查找,是指使用查找键逐个与数组元素进行比较以实现查找。其查找的基本过程为:利用循环顺序扫描整个数组,依次将每个元素与待查找值比较;若找到,则停止循环,输出其位置值;若所有元素都比较后仍未找到指定的数据值,则结束循环,输出“未找到”的提示信息。

代码实现

package com.datestructures.search;

public class SeqSearch {
    //线性查找
    //遍历元素列表  找到目标值
    public static void main(String[] args) {
        int[] arr = {2,5,3,18,7,-3,9,4,-1,0};
        int index = seqSearch(arr,7);
        if(index!=-1){
            System.out.println("找到目标值,其索引为"+index);
        }else{
            System.out.println("没有找到目标值");
        }
    }

    /**
     * 找到则返回索引  没找到 返回-1
     * @param arr
     * @param value
     * @return
     */
    public static int seqSearch(int[] arr,int value){
        for (int i = 0; i < arr.length; i++) {
            if(arr[i]==value){
                return i;
            }
        }
        return -1;
    }
}

2.二分查找算法

参考:二分法查找算法_Land-Cruise的博客-CSDN博客

代码实现

package com.datestructures.search;

import java.util.ArrayList;

public class BinarySearch {
    //二分查找  列表必须有序
    public static void main(String[] args) {
        int[] arr = {-5, -3, -1, 4, 6, 6, 6, 6, 6, 8, 11, 33, 62};
        //第一种方式查询
        int index = binarySearch(arr, 0, arr.length - 1, 6);
        //第二种方式查询
        ArrayList<Integer> list = binarySearch1(arr, 0, arr.length - 1, 6 );
        System.out.println(index);
        System.out.println(list);
    }

    public static int binarySearch(int[] arr, int left, int right, int value) {
        //这个方法只能返回一个  但是如果有几个相同的值那怎么办  所以升级一下
        if (left > right) {
            return -1;//当left大于right时,就遍历完整个数组还没有找到目标值  返回-1
        }
        int mid = (left + right) / 2;
        int midValue = arr[mid];
        if (value > midValue) {
            return binarySearch(arr, mid + 1, right, value);//向右递归
        } else if (value < midValue) {
            return binarySearch(arr, left, mid - 1, value);//向左递归
        } else {
            return mid;
        }
    }

    public static ArrayList<Integer> binarySearch1(int[] arr, int left, int right, int value) {
        //这个方法只能返回一个  但是如果有几个相同的值那怎么办  所以升级一下
        if (left > right) {
            return new ArrayList<Integer>();//当left大于right时,就遍历完整个数组还没有找到目标值  返回-1
        }
        int mid = (left + right) / 2;
        int midValue = arr[mid];
        if (value > midValue) {
            return binarySearch1(arr, mid + 1, right, value);//向右递归
        } else if (value < midValue) {
            return binarySearch1(arr, left, mid - 1, value);//向左递归
        } else {
            int temp = mid - 1;
            ArrayList<Integer> list = new ArrayList<>();
            //向左遍历看是否有相同元素
            while (true) {
                if (temp < 0 || arr[temp] != value) {
                    break;
                }
                list.add(temp);
                temp = temp - 1;
            }
            list.add(mid);
            //向右遍历看是否有相同元素
            temp = mid + 1;
            while (true) {
                if (temp > arr.length - 1 || arr[temp] != value) {
                    break;
                }
                list.add(temp);
                temp = temp + 1;
            }
            return list;
        }
    }
}

3.插值查找算法

介绍:插值查找(Interpolation Search)是根据要查找关键字key与查找表中最大最小记录的关键字比较后的查找方法,其核心就在于插值的计算公式findVal-arr[left]/arr[right]-arr[left]。细看是不是findVal在整序列中的占比哟。所以mid的计算公式为:

mid = left + (right - left) * (fidVal - arr[left]) / (arr[right] - arr[left])。

代码实现

package com.datestructures.search;

import java.io.InputStreamReader;
import java.util.Arrays;

public class InsertionValueSearch {
    //插值查找
    //适用于均值分布均匀的查找
    public static void main(String[] args) {
        int[] arr = new int[100];
        for (int i = 1; i <= 100; i++) {
            arr[i-1] = i;
        }
        System.out.println(Arrays.toString(arr));
        System.out.println(InsertionValueSearch(arr,0, arr.length-1,1 ));
    }

    //插值查找算法
    //分布不均匀用二分查找效果好
    /**
     * @param arr    要查找的数组
     * @param left   左边索引
     * @param right  右边索引
     * @param fidVal 要查找的值
     * @return 如果找到返回索引,没有找到返回-1
     */
    public static int InsertionValueSearch(int[] arr, int left, int right, int fidVal) {
        //防止数组越界  使数据有效
        if (left > right || fidVal < arr[0] || fidVal > arr[arr.length - 1]) {
            return -1;
        }
        //求mid  自适应
        int mid = left + (right - left) * (fidVal - arr[left]) / (arr[right] - arr[left]);
        if(fidVal>arr[mid]){//说明应该向右递归查找
            return InsertionValueSearch(arr,mid+1, right,fidVal);
        }else  if(fidVal<arr[mid]){//说明应该向左递归查找
            return InsertionValueSearch(arr,left,mid-1,fidVal);
        }else {
            return mid;
        }
    }
}

4.斐波那契查找算法

介绍:“斐波那契查找原理与二分查找相似,仅仅改变了中间结点(mid)的位置,mid不再是中间或插值得到,而是位于黄金分割点附近。

黄金分割点
我们知道,斐波那契数列 F(k)={1,1,2,3,5,8,13,21…},(从第三个数开始,后边每一个数都是前两个数的和)。
F[k] = F[k-1] + F[k-2]
恰巧斐波那契数列前一项/后一项的值越来越接近黄金比例,即0.618。

前提要求
待查找表为有序表。。
它要求开始表中记录的个数为某个斐波那契数小1,即n=F(k)-1。

代码实现

package com.datestructures.search;

import java.util.Arrays;

public class FibonacciSearch {
    //斐波那锲查找算法
    public static int maxSize = 20;
    public static void main(String[] args) {
        int[] arr = {1,8,10,89,1000,1234};
        System.out.println("index="+FibonacciSearch(arr,1234));
    }

    //得到一个斐波那锲数组
    public static int[] fib(){
        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 arr  数组
     * @param key  要查找的值
     * @return 如果存在返回索引,如果不存在,返回-1
     */
    public static int FibonacciSearch(int[] arr,int key){
        int low = 0;
        int high = arr.length-1;
        int k = 0;//表示斐波那锲分割数值下标
        int mid = 0;//存放mid的值
        int[] f = fib();//获取斐波那契数列
        //第一步先获取斐波那契数列分割数值的下标
        while (high>f[k]-1){
            k++;
        }
        //因为f[k]的值可能大于arr的长度 所以我们需要重新构造一个Arrays类
        int[] temp = Arrays.copyOf(arr, f[k]);//不足会用0填充
        //我们将arr[high]填充
        for (int i = high+1; i < temp.length; i++) {
            temp[i] = arr[high];
        }
        //使用循环来处理数组,来寻找我们的数值下标
        while (low<=high){
            mid = low+f[k-1]-1;//f[k] = f[k-1]+f[k-2]  f[k-1]就相当于mid  所以mid=low+f[k-1]-1;
            if(key<temp[mid]){//继续向左查找
                high = mid -1;
                k--;//相当于在前面的基础上再次分割成斐波那契数列  然后将mid  指向分割后的黄金分割点
            }else if(key>temp[mid]){//继续向右查找
                low = mid + 1;
                k-=2;//相当于在前面的基础上再将后面的分割成斐波那锲数列,然后将mid  指向分割后的黄金分割点
            }else{//找到  需要确定是哪个下标
                 if(mid<=high){
                     return mid;
                 }else{
                     return high;
                 }
            }
        }
        return -1;//没有找到返回-1;
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值