一 插值查找原理
插值查找算法类似于二分查找,不同的是插值查找每次从自适应 mid 处开始查找。
二 插值查找公式
将二分查找中的求 mid 索引的公式进行改造 , low 表示左边索引 left, high 表示右边索引 right,key 就是 findVal。
三 插值公式转代码
1 插值公式
int mid = low + (high - low) * (key - arr[low]) / (arr[high] - arr[low]) ;
2 插值公式转代码
int mid = left + (right – left) * (findVal – arr[left]) / (arr[right] – arr[left])
四 示例
数组为 [1, 2, 3, ......., 100]
1 假如需要查找的值为1
使用二分查找的话,需要多次递归,才能找到1。
使用插值查找算法
int mid = left + (right – left) * (findVal – arr[left]) / (arr[right] – arr[left])
int mid = 0 + (99 - 0) * (1 - 1)/ (100 - 1) = 0 + 99 * 0 / 99 = 0
一次查找即可。
2 假如需要查找的值为100
使用插值查找算法
int mid = 0 + (99 - 0) * (100 - 1) / (100 - 1) = 0 + 99 * 99 / 99 = 0 + 99 = 99
一次查找即可。
五 实战
1 要求
请对一个有序数组进行插值查找 {1,8, 10, 89, 1000, 1234} ,输入一个数看看该数组是否存在此数,并且求出下标,如果没有就提示"没有这个数"。
2 代码
/**
* @className: InsertValueSearch
* @description: 插值查找
* @date: 2021/3/13
* @author: cakin
*/
public class InsertValueSearch {
/**
* 功能描述:插值查找测试
*
* @param args 命令行
* @author cakin
* @date 2021/3/13
*/
public static void main(String[] args) {
// 待查找数组
int arr[] = {1, 8, 10, 89, 1000, 1000, 1234};
int index = insertValueSearch(arr, 0, arr.length - 1, 1234);
System.out.println("index = " + index);
}
/**
* 功能描述: 插值查找算法。插值查找算法,也要求数组是有序的
*
* @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;
}
}
}
3 测试
插值查找次数
index = 6
4 说明
效率很高,查找一次就找到了。