查找算法
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;
}
}