二分查找入门详解

在平常的软件开发过程中,我们经常都会遇到需要在数组或集合中查找某个指定元素的情况。通常情况下,我们会使用按照自然顺序的方式来查找数组中的是否存在指定的元素。例如:

 
  1. /**
  2.  * 从数组中顺序查找是否存在指定的元素,如果存在则返回该元素的索引,否则返回-1
  3.  *
  4.  * @param array 指定所查找的数组
  5.  * @param search 待查找的元素
  6.  * @return
  7.  */
  8. public static final int search(int[] array, int search) {
  9.     for (int i = 0; i < array.length; i++) {
  10.         if (array[i] == search) {
  11.             return i;
  12.         }
  13.     }
  14.     return -1;
  15. }

不过,作为程序员更应该掌握能够提高查找效率的相关算法,比如——二分查找(binary search)二分查找是编程技术领域一种非常著名也比较简单的算法,它可以使得查找元素的效率得到几何量级的提高。不过,值得注意的是,二分查找的使用前提是:所查找的排列必须是一个有序(从大到小或从小到大)的排列

二分查找算法,简单点说,就是先将一个有序排列最中间的元素x取出来与查找的元素进行比较,从而确定查找的元素是在x的左侧还是右侧(或者是x本身),从而缩小查找的区间范围。然后再继续取出缩小范围后的有序排列中最中间的元素与查找的元素进行比较,再次确定查找的元素是在其左侧还是右侧,通过多次这样的循环筛选,不断缩小查找的区间范围,最后确定待查找元素的位置。

如上所述,二分查找算法是一种比较简单的算法。不过,遗憾的是,美国著名的编程技术专家乔恩·本特利(Jon Bentley)在1986年的经典名著《编程珠玑》(Programming Pearls)中提到他的一个调查结果:居然有90%左右的程序员无法在2小时内使用自己喜欢的编程语言正确编写出二分查找的代码。因此,我们不得不在这里老生常谈似的,再次回顾一下二分查找算法代码的实现:

 
  1. /**
  2.  * 使用二分查找算法在有序数组中查找指定的元素,如果找到则返回对应的数组索引,否则返回-1
  3.  * @param array 待查找的有序数组(注意,是有序数组!)
  4.  * @param search 需要查找的元素
  5.  * @return
  6.  */
  7. public static final int binarySearch(int[] array, int search) {
  8.     int start = 0, end = array.length - 1;
  9.     while (start <= end) {
  10.         int mid = (start + end) / 2;
  11.         if (array[mid] < search) {
  12.             start = mid + 1;
  13.         } else if (array[mid] > search) {
  14.             end = mid - 1;
  15.         } else {
  16.             return mid;
  17.         }
  18.     }
  19.     return -1;
  20. }
  21.  
  22. // 程序主方法
  23. public static void main(String[] args) {
  24.     int[] array = { 0, 12, 25, 38, 49, 65, 81 }; //必须是事先已经排序好的数组
  25.     System.out.println(binarySearch(array, 65)); // 输出:5
  26. }

上述代码也是网络上其他介绍二分查找算法的文章中给出的完整代码实现,可是这样就结束了吗?No,当然不是,这还不是完整的二分查找实现代码。众所周知,二分查找是针对一个有序排列而设计的数组查找算法,既然是有序排列,不仅从小到大的排列是有序排列,从大到小的排列也是有序排列。但是,上述代码却只能实现对从小到大的有序排列的二分查找,如果是从大到小的有序排列,它就无能为力了。因此,我们需要针对从大到小的有序排列的情况来完善上述代码:

 
  1. /**
  2.  * 使用二分查找算法在有序数组中查找指定的元素,如果找到则返回对应的数组索引,否则返回-1
  3.  *
  4.  * @param array 待查找的有序数组(注意,是有序数组!)
  5.  * @param search 需要查找的元素
  6.  * @return
  7.  */
  8. public static final int binarySearch(int[] array, int search) {
  9.     int start = 0, end = array.length - 1;
  10.     // 判断当前数组为正向排序(从小到大)还是逆向排序(从大到小)
  11.     boolean isPositive = array[end] >= array[start];
  12.     while (start <= end) {
  13.         int mid = (start + end) / 2;
  14.         if (array[mid] < search) {
  15.             if (isPositive) {
  16.                 start = mid + 1;
  17.             } else {
  18.                 end = mid - 1;
  19.             }
  20.         } else if (array[mid] > search) {
  21.             if (isPositive) {
  22.                 end = mid - 1;
  23.             } else {
  24.                 start = mid + 1;
  25.             }
  26.         } else {
  27.             return mid;
  28.         }
  29.     }
  30.     return -1;
  31. }
  32.  
  33. // 程序主方法
  34. public static void main(String[] args) {
  35.     int[] array = { 81, 65, 49, 38, 25, 12, 0 };
  36.     System.out.println(binarySearch(array, 65)); // 输出:1
  37. }

备注:真正完整的二分查找算法实现应该兼容多种数据类型,为了便于读者阅读理解,上述代码仅实现了针对int类型的有序数组的二分查找。

你可能感兴趣的:(java算法)