Java使用二分法进行查找和排序的示例

 更新时间:2016年04月09日 08:51:22   作者:匆忙拥挤repeat  
这篇文章主要介绍了Java使用二分法进行查找和排序的示例,二分插入排序和二分查找是基础的算法,需要的朋友可以参考下

实现二分法查找
二分法查找,需要数组内是一个有序的序列
二分查找比线性查找:数组的元素数越多,效率提高的越明显
二分查找的效率表示:O(log2N) N在2的M次幂范围,那查找的次数最大就是M,  log2N表示2的M次幂等于N, 省略常数,简写成O(logN)
如有一个200个元素的有序数组,那么二分查找的最大次数:
2^7=128, 2^8=256, 可以看出7次幂达不到200,8次幂包括, 所以最大查找次数就等于8

//循环,二分查找

static int binarySearch(int[] array, int data) { 
  int start = 0; 
  int end = array.length - 1; 
  int mid = -1; 
  while (start <= end) { 
   System.out.println("查找次数"); 
   mid = (start + end) >>> 1; 
   if (array[mid] < data) { 
    start = mid + 1; 
   } else if (array[mid] > data) { 
    end = mid - 1; 
   } else { 
    return mid; 
   } 
   System.out.println("start=" + start+",end="+end+",mid="+mid); 
  } 
  return -1; 
 } 

//递归二分查找 初始start=0, end = array.length - 1 
 static int binarySearch4Recursion(int[] array, int data, int start, int end) { 
  int mid = -1; 
  System.out.println("查找次数"); 
  if (start > end) { 
   return mid; 
  } 
  mid = (start + end) >>> 1; 
  if (array[mid] < data) { 
   return binarySearch4Recursion(array, data, mid + 1, end); 
  } else if (array[mid] > data) { 
   return binarySearch4Recursion(array, data, start, mid - 1); 
  } else { 
   return mid; 
  } 
    
 } 

二分法插入排序

设有一个序列a[0],a[1]...a[n];其中a[i-1]前是已经有序的,当插入时a[i]时,利用二分法搜索a[i]插入的位置
效率:O(N^2),对于初始基本有序的序列,效率上不如直接插入排序;对于随机无序的序列,效率比直接插入排序要高

/* 
 * 二分(折半)插入排序 
 * 设有一个序列a[0],a[1]...a[n];其中a[i-1]前是已经有序的,当插入时a[i]时,利用二分法搜索a[i]插入的位置 
 */ 
public class BinaryInsertSort { 
 
 public static void main(String[] args) { 
  int len = 10; 
  int[] ary = new int[len]; 
  Random random = new Random(); 
  for (int j = 0; j < len; j++) { 
   ary[j] = random.nextInt(1000); 
  } 
  binaryInsert(ary); 
  /* 
   * 复杂度分析: 最佳情况,即都已经排好序,则无需右移,此时时间复杂度为:O(n lg n) 最差情况,全部逆序,此时复杂度为O(n^2) 
   * 无法将最差情况的复杂度提升到O(n|logn)。 
   */ 
  // 打印数组 
  printArray(ary); 
 } 
 /** 
  * 插入排序 
  * @param ary 
  */ 
 private static void binaryInsert(int[] ary) { 
  int setValueCount = 0; 
  // 从数组第二个元素开始排序,因为第一个元素本身肯定是已经排好序的 
  for (int j = 1; j < ary.length; j++) {// 复杂度 n 
   // 保存当前值 
   int key = ary[j]; 
   // ∆ 利用二分查找定位插入位置 
//   int index = binarySearchAsc(ary, ary[j], 0, j - 1);// 复杂度:O(logn) 
//   int index = binarySearchDesc(ary, ary[j], 0, j - 1);// 复杂度:O(logn) 
   int index = binarySearchDesc2(ary, ary[j], 0, j - 1);// 复杂度:O(logn) 
   printArray(ary); 
   System.out.println("第" + j +"个索引上的元素要插入的位置是:" + index); 
   // 将目标插入位置,同时右移目标位置右边的元素 
   for (int i = j; i > index; i--) {// 复杂度,最差情况:(n-1)+(n-2)+...+n/2=O(n^2) 
    ary[i] = ary[i - 1]; //i-1 <==> index 
    setValueCount++; 
   } 
   ary[index] = key; 
   setValueCount++; 
  } 
  System.out.println("\n 设值次数(setValueCount)=====> " + setValueCount); 
 } 
 
 /** 
  * 二分查找 升序 递归 
  * 
  * @param ary 
  *   给定已排序的待查数组 
  * @param target 
  *   查找目标 
  * @param from 
  *   当前查找的范围起点 
  * @param to 
  *   当前查找的返回终点 
  * @return 返回目标在数组中,按顺序应在的位置 
  */ 
 private static int binarySearchAsc(int[] ary, int target, int from, int to) { 
  int range = to - from; 
  // 如果范围大于0,即存在两个以上的元素,则继续拆分 
  if (range > 0) { 
   // 选定中间位 
   int mid = (to + from) / 2; 
   // 如果临界位不满足,则继续二分查找 
   if (ary[mid] > target) { 
    /* 
     * mid > target, 升序规则,target较小,应交换位置 前置, 即target定位在mid位置上, 
     * 根据 查找思想, 从from到 mid-1认为有序, 所以to=mid-1 
     */ 
    return binarySearchAsc(ary, target, from, mid - 1); 
   } else { 
    /* 
     * mid < target, 升序规则,target较大,不交换位置,查找比较的起始位置应为mid+1 
     */ 
    return binarySearchAsc(ary, target, mid + 1, to); 
   } 
  } else { 
   if (ary[from] > target) {//如 5,4, 要插入的是4 
    return from; 
   } else { 
    return from + 1; 
   } 
  } 
 } 
 /** 
  * 二分查找 降序, 递归 
  */ 
 private static int binarySearchDesc(int[] ary, int target, int from, int to) { 
  int range = to - from; 
  if (range > 0) { 
   int mid = (from + to) >>> 1; 
   if (ary[mid] > target) { 
    return binarySearchDesc(ary, target, mid + 1, to); 
   } else { 
    return binarySearchDesc(ary, target, from, mid - 1); 
   } 
  } else { 
   if (ary[from] > target) {//如 5,4, 要插入的是4 
    return from + 1; 
   } else { 
    return from; 
   } 
  } 
 } 
  
 /** 
  * 二分查找 降序, 非递归 
  */ 
 private static int binarySearchDesc2(int[] ary, int target, int from, int to) { 
//  while(from < to) { 
  for (; from < to; ) { 
   int mid = (from + to) >>> 1; 
   if (ary[mid] > target) { 
    from = mid + 1; 
   } else { 
    to = mid -1; 
   } 
  } 
  //from <==> to; 
  if (ary[from] > target) {//如 5,4, 要插入的是4 
   return from + 1; 
  } else { 
   return from; 
  } 
 } 
 
 private static void printArray(int[] ary) { 
  for (int i : ary) { 
   System.out.print(i + " "); 
  } 
 } 
 
} 

打印

918 562 442 531 210 216 931 706 333 132 第1个索引上的元素要插入的位置是:1 
918 562 442 531 210 216 931 706 333 132 第2个索引上的元素要插入的位置是:2 
918 562 442 531 210 216 931 706 333 132 第3个索引上的元素要插入的位置是:2 
918 562 531 442 210 216 931 706 333 132 第4个索引上的元素要插入的位置是:4 
918 562 531 442 210 216 931 706 333 132 第5个索引上的元素要插入的位置是:4 
918 562 531 442 216 210 931 706 333 132 第6个索引上的元素要插入的位置是:0 
931 918 562 531 442 216 210 706 333 132 第7个索引上的元素要插入的位置是:2 
931 918 706 562 531 442 216 210 333 132 第8个索引上的元素要插入的位置是:6 
931 918 706 562 531 442 333 216 210 132 第9个索引上的元素要插入的位置是:9 

 设值次数(setValueCount)=====> 24 

931 918 706 562 531 442 333 216 210 132 

相关文章

  • 使用Spring开启@Async异步方式(javaconfig配置)

    使用Spring开启@Async异步方式(javaconfig配置)

    这篇文章主要介绍了使用Spring开启@Async异步方式(javaconfig配置),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08
  • java教程之java程序编译运行图解(java程序运行)

    java教程之java程序编译运行图解(java程序运行)

    最近重新复习了一下java基础,在使用javap的过程中遇到了一些问题,这里便讲讲对于一个类文件如何编译、运行、反编译的。也让自己加深一下印象
    2014-03-03
  • 换了最新的idea如何将原来旧版本的idea设置导进新的idea中

    换了最新的idea如何将原来旧版本的idea设置导进新的idea中

    这篇文章主要介绍了换了最新的idea如何将原来旧版本的idea设置导进新的idea中,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-11-11
  • Maven scala和java混合打包方式

    Maven scala和java混合打包方式

    这篇文章主要介绍了Maven scala和java混合打包方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-06-06
  • Java ArrayList类的基础使用讲解

    Java ArrayList类的基础使用讲解

    数组的长度是固定的,无法适应数据变化的需求。为了解决这个问题,Java提供了另一个容器 java.util.ArrayList集合类,让我们可以更便捷的存储和操作对象数据。本文就将通过示例聊聊ArrayList类的基础使用,感兴趣的可以了解一下
    2022-10-10
  • SpringBoot中的server.context-path的实现

    SpringBoot中的server.context-path的实现

    本文主要介绍了SpringBoot中的server.context-path的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-08-08
  • Java连接sftp服务器实现上传下载功能

    Java连接sftp服务器实现上传下载功能

    这篇文章主要介绍了java连接sftp服务器实现上传下载,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-08-08
  • Java操作压缩包解压过程详解

    Java操作压缩包解压过程详解

    这篇文章主要介绍了Java操作压缩包解压过程,项目开发中,总会遇到解压缩文件的时候,比如用户下载多个文件时,服务端可以将多个文件压缩成一个文件,用户上传资料时,允许上传压缩文件,服务端进行解压读取每一个文件,使用场景是很多的,下面来详细讲解,需要的朋友可以参考下
    2024-10-10
  • Spring Boot 中的 @Field 注解的原理解析

    Spring Boot 中的 @Field 注解的原理解析

    本文详细介绍了 Spring Boot 中的 @Field 注解的原理和使用方法,通过使用 @Field 注解,我们可以将 HTTP 请求中的参数值自动绑定到 Java 对象的属性上,简化了开发过程,提高了开发效率,感兴趣的朋友跟随小编一起看看吧
    2023-07-07
  • Java多线程下的单例模式参考

    Java多线程下的单例模式参考

    这篇文章主要演示多线程下的单例模式,分别演示了lock和synchronized两种方案,希望能给大家做一个参考。
    2016-06-06

最新评论