Java查找算法(线性查找,二分查找递归和非递归,插值查找,斐波那契查找算法)

查找算法

查找算法介绍

在java 中,我们常用的查找有四种:

  1. 顺序(线性)查找
  2. 二分查找/折半查找
  3. 插值查找
  4. 斐波那契查找

线性查找算法

有一个数列: {1,8, 10, 89, 1000, 1234} ,判断数列中是否包含此名称【顺序查找】要求: 如果找到了,就提示找到,并给出下标值。

代码实现:

public class SeqSearch {

	public static void main(String[] args) {
		int arr[] = { 1, 9, 11, -1, 34, 89 };// 没有顺序的数组
		int index = seqSearch(arr, -11);
		if(index == -1) {
			System.out.println("没有找到到");
		} else {
			System.out.println("找到,下标为=" + index);
		}
	}

	/**
	 * 这里我们实现的线性查找是找到一个满足条件的值,就返回
	 * @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;
	}

}

二分查找算法

二分查找:

请对一个==有序数组==进行二分查找{1,8, 10, 89, 1000, 1234} ,输入一个数看看该数组是否存在此数,并且求出下标,如果没有就提示"没有这个数"。

二分查找算法的思路

二分查找的思路分析

  1. 首先确定该数组的中间的下标
    mid = (left + right) / 2
  2. 然后让需要查找的数 findVal 和 arr[mid] 比较
    2. 1 findVal > arr[mid] , 说明你要查找的数在mid 的右边, 因此需要递归的向右查找
    2.2 findVal < arr[mid], 说明你要查找的数在mid 的左边, 因此需要递归的向左查找
    2.3 findVal == arr[mid] 说明找到,就返回

//什么时候我们需要结束递归.

  1. 找到就结束递归
  2. 递归完整个数组,仍然没有找到findVal ,也需要结束递归 当 left > right 就需要退出
    当两个重合还找不到,下一步就是left>right

二分查找的代码

说明:增加了找到所有的满足条件的元素下标:
课后思考题: {1,8, 10, 89, 1000, 1000,1234} 当一个有序数组中,有多个相同的数值时,如何将所有的数值都查找到,比如这里的1000.

import java.util.ArrayList;
import java.util.List;
//注意:使用二分查找的前提是该数组是有序的.
public class BinarySearch {
	public static void main(String[] args) {
		//一定要有序的数组
		int arr[] = { 1,8, 10, 89, 1000, 1000,1234};
		//
		// int resIndex = binarySearch(arr, 0, arr.length - 1, 1000);
		// System.out.println("resIndex=" + resIndex);
		List<Integer> resIndexList = binarySearch2(arr, 0, arr.length, 1);
		System.out.println("resIndexList=" + resIndexList);
	}

	// 二分查找算法
	/**
	 *
	 * @param arr     数组
	 * @param left    左边的索引
	 * @param right   右边的索引
	 * @param findVal 要查找的值
	 * @return 如果找到就返回下标,如果没有找到,就返回-1
	 */
	public static int binarySearch(int[] arr, int left, int right, int findVal) {
		// 当left > right 时,说明递归整个数组,但是没有找到
		if (left > right) {
			return -1;
		}
		int mid = (left + right) / 2;
		int midVal = arr[mid];
		if (findVal > midVal) { // 向右递归
			return binarySearch(arr, mid + 1, right, findVal);
		} else if (findVal < midVal) { // 向左递归
			return binarySearch(arr, left, mid - 1, findVal);
		} else {
			return mid;
		}
	}

	// 完成一个课后思考题:
	/*
	 * 课后思考题: {1,8, 10, 89, 1000, 1000,1234} 当一个有序数组中,
	 * 有多个相同的数值时,如何将所有的数值都查找到,比如这里的1000
	 *
	 * 思路分析 1. 在找到mid 索引值,不要马上返回 2. 向mid 索引值的左边扫描,将所有满足1000, 的元素的下标,加入到集合ArrayList
	 * 3. 向mid 索引值的右边扫描,将所有满足1000, 的元素的下标,加入到集合ArrayList 4. 将Arraylist 返回
	 */
	public static List<Integer> binarySearch2(int[] arr, int left, int right, int findVal) {
		// 当left > right 时,说明递归整个数组,但是没有找到
		if (left > right) {
			return new ArrayList<Integer>();
		}
		int mid = (left + right) / 2;
		int midVal = arr[mid];
		if (findVal > midVal) { // 向右递归
			return binarySearch2(arr, mid + 1, right, findVal);
		} else if (findVal < midVal) { // 向左递归
			return binarySearch2(arr, left, mid - 1, findVal);
		} else {
			// * 思路分析
			// * 1. 在找到mid 索引值,不要马上返回
			// * 2. 向mid 索引值的左边扫描,将所有满足1000, 的元素的下标,加入到集合ArrayList
			// * 3. 向mid 索引值的右边扫描,将所有满足1000, 的元素的下标,加入到集合ArrayList
			// * 4. 将Arraylist 返回
			List<Integer> resIndexlist = new ArrayList<Integer>();
			// 向mid 索引值的左边扫描,将所有满足1000, 的元素的下标,加入到集合ArrayList
			int temp = mid - 1;
			while (true) {
				if (temp < 0 || arr[temp] != findVal) {// 退出
					break;
				}
				// 否则,就temp 放入到resIndexlist
				resIndexlist.add(temp);
				temp -= 1; // temp 左移
			}
			resIndexlist.add(mid); //记得还要放mid
			// 向mid 索引值的右边扫描,将所有满足1000, 的元素的下标,加入到集合ArrayList
			temp = mid + 1;
			while (true) {
				if (temp > arr.length - 1 || arr[temp] != findVal) {// 退出
					break;
				}
				// 否则,就temp 放入到resIndexlist
				resIndexlist.add(temp);
				temp += 1; // temp 右移
			}
			return resIndexlist;
		}
	}
}

二分查找算法(非递归)面试高频

二分查找算法(非递归)介绍

  1. 前面我们讲过了二分查找算法,是使用递归的方式,下面我们讲解二分查找算法的非递归方式
  2. 二分查找法只适用于从有序的数列中进行查找(比如数字和字母等),将数列排序后再进行查找
  3. 二分查找法的运行时间为对数时间O(㏒₂n) ,即查找到需要的目标位置最多只需要㏒₂n 步,假设从[0,99]的队列(100 个数,即n=100)中寻到目标数30,则需要查找步数为㏒₂100 , 即最多需要查找7 次( 2^6 < 100 < 2^7)

二分查找算法(非递归)代码实现

数组{1,3, 8, 10, 11, 67, 100}, 编程实现二分查找, 要求使用非递归的方式完成.

	public static void main(String[] args) {
		// 测试
		int[] arr = { 1, 3, 8, 10, 11, 67, 100 };
		int index = binarySearch(arr, 100);
		System.out.println("index=" + index);//
	}

	// 二分查找的非递归实现
	/**
	 * 
	 * @param arr    待查找的数组, arr是升序排序
	 * @param target 需要查找的数
	 * @return 返回对应下标,-1表示没有找到
	 */
	public static int binarySearch(int[] arr, int target) {

		int left = 0;
		int right = arr.length - 1;
		while (left <= right) { // 说明继续查找
			int mid = (left + right) / 2;
			if (arr[mid] == target) {
				return mid;
			} else if (arr[mid] > target) {
				right = mid - 1;// 需要向左边查找
			} else {
				left = mid + 1; // 需要向右边查找
			}
		}
		return -1;
	}

}

插值查找算法

插值查找原理介绍

  1. 插值查找原理介绍:
    插值查找算法类似于二分查找,不同的是插值查找每次从自适应mid 处开始查找。
    (自适应)要求的值本身也参与了mid的等式
  2. 将折半查找中的求mid 索引的公式, low 表示左边索引left, high 表示右边索引right.
    在这里插入图片描述
    key 就是前面我们讲的findVal
    int mid = low + (high - low) * (key - arr[low]) / (arr[high] - arr[low]) ;/插值索引/
    对应前面的代码公式:
    int mid = left + (right – left) * (findVal – arr[left]) / (arr[right] – arr[left])
  3. 举例说明插值查找算法1-100 的数组
    在这里插入图片描述

插值查找应用案例:

请对一个有序数组进行插值查找{1,8, 10, 89, 1000, 1234} ,输入一个数看看该数组是否存在此数,并且求出下标,如果没有就提示"没有这个数"。
代码实现:

import java.util.Arrays;

public class InsertValueSearch {

	public static void main(String[] args) {
		
//		int [] arr = new int[100];
//		for(int i = 0; i < 100; i++) {
//			arr[i] = i + 1;
//		}
		
		int arr[] = { 1, 8, 10, 89,1000,1000, 1234 };
		
		int index = insertValueSearch(arr, 0, arr.length - 1, 89);
		System.out.println("index = " + index);
		//System.out.println(Arrays.toString(arr));
	}
	
	

	//编写插值查找算法
	//说明:插值查找算法,也要求数组是有序的
	/**
	 * 
	 * @param arr 数组
	 * @param left 左边索引
	 * @param right 右边索引
	 * @param findVal 查找值
	 * @return 如果找到,就返回对应的下标,如果没有找到,返回-1
	 */
	public static int insertValueSearch(int[] arr, int left, int right, int findVal) { 

		System.out.println("插值查找次数~~");
		
		//注意:findVal < arr[0]  和  findVal > arr[arr.length - 1] 必须需要
		//否则我们得到的 mid 可能越界
		if (left > right || findVal < arr[0] || findVal > arr[arr.length - 1]) {
			return -1;
		}

		// 求出mid, 自适应
		int mid = left + (right - left) * (findVal - arr[left]) / (arr[right] - arr[left]);
		int midVal = arr[mid];
		if (findVal > midVal) { // 说明应该向右边递归
			return insertValueSearch(arr, mid + 1, right, findVal);
		} else if (findVal < midVal) { // 说明向左递归查找
			return insertValueSearch(arr, left, mid - 1, findVal);
		} else {
			return mid;
		}

	}
}

插值查找注意事项:

  1. 对于数据量较大,关键字分布比较均匀(适合连续性的)的查找表来说,采用插值查找, 速度较快.
  2. 关键字分布不均匀数组里面的值的跳跃性很大)的情况下,该方法不一定比折半查找要好

问题

插值查找为什么是( findvalue-arr[1eft])/( arr[right]-arr1ef])?

  • 首先我们根据基于数组的有序性,想一想查找的位置一定要从中间开始查找吗?

    打个比方:我们在一本英文字典里面查找aple这个单词的时候,你肯定不会从字典中间开始查找,而是从字典开头部分开始翻,因为觉得这样的找法才是比较快的

    那么为什么你会从头部分开始翻呢?因为知道字典是按照字母有序排序的,并且 apple这个单词是a开头,所以从头开始翻,对嘛?

    那么这时就有了一个思路:如果能在查找前较准确地预测关键字在数组中的位置的话,这样的查找方法能比分查找提高更多的性能!

    所以根据差值公式(key-arr1ow])/( arr
    [high]-ar1ow]),将要查找的关键字key与查找表中的最大、最小记录的关键字比较的查找方法

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

斐波那契(黄金分割法)查找基本介绍:

  1. 黄金分割点是指把一条线段分割为两部分,使其中一部分与全长之比等于另一部分与这部分之比。取其前三位数字的近似值是0.618。由于按此比例设计的造型十分美丽,因此称为黄金分割,也称为中外比。这是一个神奇的数字,会带来意向不大的效果。
  2. 斐波那契数列{1, 1, 2, 3, 5, 8, 13, 21, 34, 55 } 发现斐波那契数列的两个相邻数的比例,无限接近黄金分割值0.618

斐波那契(黄金分割法)原理:

F(k-1)/f(K)随着K的递增,该数越来越接近黄金分割比例

斐波那契查找原理与前两种相似,仅仅改变了中间结点(mid)的位置,mid 不再是中间或插值得到,而是位于黄金分割点附近,即mid=low+F(k-1)-1(F 代表斐波那契数列),如下图所示
在这里插入图片描述
对F(k-1)-1 的理解:

  1. 由斐波那契数列F[k]=F[k-1]+F[k-2] 的性质,可以得到(F[k]-1)=(F[k-1]-1)+(F[k-2]-1)+1 。该式说明:

只要顺序表的长度为F[k]-1,则可以将该表分成长度为F[k-1]-1 和F[k-2]-1 的两段,即如上图所示。从而中间位置为mid=low+F(k-1)-1

  1. 类似的,每一子段也可以用相同的方式分割
  2. 但顺序表长度n 不一定刚好等于F[k]-1,所以需要将原来的顺序表长度n 增加至F[k]-1。这里的k 值只要能使得F[k]-1 恰好大于或等于n 即可,由以下代码得到,顺序表长度增加后,新增的位置(从n+1 到F[k]-1 位置),都赋为n 位置的值即可。
    while(n>fib(k)-1)
    k++;

原理2

  1. 斐波那契查找也叫做黄金分割法查找,它也是根据折半查找算法来进行修改和改进的。
  2. 对于斐波那契数列:1、1、2、3、5、8、13、21、34、55、89……(也可以从0开始),前后两个数字的比值随着数列的增加,越来越接近黄金比值0.618。比如这里的89,把它想象成整个有序表的元素个数,而89是由前面的两个斐波那契数34和55相加之后的和,也就是说把元素个数为89的有序表分成由前55个数据元素组成的前半段和由后34个数据元素组成的后半段,那么前半段元素个数和整个有序表长度的比值就接近黄金比值0.618,假如要查找的元素在前半段,那么继续按照斐波那契数列来看,55 = 34 + 21,所以继续把前半段分成前34个数据元素的前半段和后21个元素的后半段,继续查找,如此反复,直到查找成功或失败,这样就把斐波那契数列应用到查找算法中了。如下图:
    在这里插入图片描述
    从图中可以看出,当有序表的元素个数不是斐波那契数列中的某个数字时,需要把有序表的元素个数长度补齐,让它成为斐波那契数列中的一个数值,当然把原有序表截断肯定是不可能的,不然还怎么查找。然后图中标识每次取斐波那契数列中的某个值时(F[k]),都会进行-1操作,这是因为有序表数组位序从0开始的,纯粹是为了迎合位序从0开始。

原理2来自:https://blog.csdn.net/cdu09/article/details/23122623

斐波那契查找应用案例:

请对一个==有序数组==进行斐波那契查找{1,8, 10, 89, 1000, 1234} ,输入一个数看看该数组是否存在此数,并且求出下标,如果没有就提示"没有这个数"。
代码实现:

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=" + fibSearch(arr, 189));// 0
		
	}

	//因为后面我们mid=low+F(k-1)-1,需要使用到斐波那契数列,因此我们需要先获取到一个斐波那契数列
	//非递归方法得到一个斐波那契数列
	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 a  数组
	 * @param key 我们需要查找的关键码(值)
	 * @return 返回对应的下标,如果没有-1
	 */
	public static int fibSearch(int[] a, int key) {
		int low = 0;
		int high = a.length - 1;
		int k = 0; //表示斐波那契分割数值的下标
		int mid = 0; //存放mid值
		int f[] = fib(); //获取到斐波那契数列
		//获取到斐波那契分割数值的下标
		while(high > f[k] - 1) {
			k++;
		}
		//因为 f[k] 值 可能大于 a 的 长度,因此我们需要使用Arrays类,构造一个新的数组,并指向temp[]
		//不足的部分会使用0填充
		int[] temp = Arrays.copyOf(a, f[k]);
		//实际上需求使用a数组最后的数填充 temp
		//举例:
		//temp = {1,8, 10, 89, 1000, 1234, 0, 0}  => {1,8, 10, 89, 1000, 1234, 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-1] = f[k-3] + f[k-4]
				//4. 即在f[k-2] 的前面进行查找 k -=2
				//5. 即下次循环 mid = f[k - 1 - 2] - 1
				k -= 2;
			} else { //找到
				//需要确定,返回的是哪个下标
				if(mid <= high) {
					return mid;
				} else {
					return high;
				}
			}
		}
		return -1;
	}
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值