二分查找算法及其变种
来源:互联网 发布:android 离线数据缓存 编辑:程序博客网 时间:2024/05/16 05:33
前言
二分查找算法也称为折半查找算法,是一种在查找算法中普遍使用的算法。其算法的基本思想是:在有序表中,取中间的记录作为比较关键字,若给定值与中间记录的关键字相等,则查找成功;若给定的值小于中间记录的关键字,则在中间记录的左半区间继续查找;若给定值大于中间记录的关键字,则在中间记录的右半区间继续查找;不断重复这个过程,直到查找成功。否则查找失败。这个思想与孔子中的中庸思想和相似。
二分查找算法的实现
基于上述的思想,可以很快写出如下代码:
public int binarySearch(int[] a,int key) { int low = 0; int high = a.length - 1; int mid = 0; while(low <= high){ mid = (low + high) / 2; if(a[mid] == key) return mid; if(a[mid] > key) high = mid - 1; if(a[mid] < key) low = mid + 1; } return -1; }
实际上,二分查找的过程可以绘制成一棵二叉树,每次二分查找的过程就相当于把原来的树划分为两棵子树,所以每次二分之后下次就只需要查找其中一半的数据就可以了。那么二分查找算法的时间复杂度是多少呢?在最好的情况下,只需要查找一次就可以了,因为这时候中间记录的关键字与要查找的key是相等,自然一次就够了。在最坏的情况下是从根节点查找到最下面的叶子结点,这个过程需要的时间复杂度是
需要注意的是,虽然二分查找算法的效率很高(这也是二分查找算法被广泛应用的原因),但是仍然是有使用条件的:有序。就是说在需要频繁进行插入或者删除操作的数据记录中使用二分查找算法不太划算,因为要维持数据的有序还需要额外的排序开销。
二分查找算法的变种一:插值查找算法
可以发现二分查找每次都是选取中间的那个记录关键字作为划分依据的,那为什么不可以是其他位置的关键字呢?在有些情况下,使用二分查找算法并不是最合适的。举个例子:在1-1000中,一共有1000个关键字,如果要查找关键字10,按照二分查找算法,需要从500开始划分,这样的话效率就比较低了,所以有人提出了插值查找算法。说白了就是改变划分的比例,比如三分或者四分。
插值查找算法对二分查找算法的改进主要体现在mid的计算上,其计算公式如下:
而原来的二分查找公式是这样的:
所以我们发现主要变化的地方是
二分查找算法变种二:斐波那契查找算法
从前面的分析中可以看到,无论划分的关键字太大或者太小都不合适,所以又有人提出了斐波那契查找算法,其利用了黄金分割比原理来实现的。
一个数列如果满足F(n)=F(n-1)+F(n-2),则称这个数列为斐波那契数列。在斐波那契查找算法中计算mid的公式如下:
其实现代码如下:
package com.rhwayfun.algorithm.search;public class FibonacciSearch { public int fibonacciSearch(int[] a,int key){ int low = 0,high = a.length - 1,mid = 0,k = 0,i =0; //计算数组的长度的值在斐波那契数列的位置 while(a.length > F(k) - 1){ k++; } //将不满的数值补全 int[] newArray = new int[F(k) - 1]; System.arraycopy(a, 0, newArray, 0, a.length); for(i = a.length; i < F(k) - 1; i++) newArray[i] = a[a.length - 1]; a = newArray; //查找过程 while(low <= high){ mid = low + F(k-1) - 1; if(key < a[mid]){ high = mid - 1; k = k - 1; }else if(key > a[mid]){ low = mid + 1; k = k - 2; }else{ if(mid < a.length){ return mid; }else{ //说明是补全之后的数值 return a.length - 1; } } } return 0; } //返回第n项斐波那契数列的值 private int F(int n) { if(n == 0){ return 0; }else if(n == 1){ return 1; } int one = 1; int two = 0; int sum = 0; for (int i = 2; i <= n; i++) { sum = one + two; two = one; one = sum; } return sum; } public static void main(String[] args) { int[] a = {0,1,16,24,35,47,59,62,73,88,99}; int i = new FibonacciSearch().fibonacciSearch(a, 59); System.out.println(a[i]); }}
可以看出斐波那契查找算法的核心是如果要查找的记录在右侧,则左边就不会再去查找了,不断反复进行下去,知道查找成功。虽然斐波那契查找算法的时间复杂度也是
- 二分查找算法及其变种
- 二分查找算法及其变种
- 二分查找及其变种总结
- 数据结构与算法之二分查找法及其变种
- 算法 二分查找的变种以及注意事项
- 变种二分查找
- 二分查找及变种
- 二分查找的变种
- 二分查找及其变形算法
- 快速排序及其变种算法
- c语言实现二分查找变种题型
- 算法系列——二分查找算法及其变体总结
- 3.18二分搜索算法各种变种
- QuickSort及其变种算法的总结
- 约瑟夫环及其变种算法 java实现
- 二分查找及其变形
- 二分查找及其应用
- 二分查找及其变体
- Launcher3 桌面壁纸滑动
- 最近邻算法(KNN)
- 抓取维基百科数据
- 【插件开发二】开发OpenFire的消息拦截器
- 非递归实现二叉树遍历(思路+代码)
- 二分查找算法及其变种
- Python中用encoding声明的文件编码和文件的实际编码之间的关系
- ViewPager Adapter FragmentStatePagerAdapter FragmentPagerAdapter 使用详解
- 迭代器
- POJ 1177 Picture(线段树+扫描线)
- Eclipse OS X Yosemite 提示– To open“Eclipse” you need to install the legacy Java SE 6 runtime的解决办法
- pdo+mysql+php
- mybatis实战教程(mybatis in action),mybatis入门到精通
- saltstack相关的一些总结