在路上

 找回密码
 立即注册
在路上 站点首页 学习 查看内容

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

2016-7-29 15:41| 发布者: zhangjf| 查看: 650| 评论: 0

摘要: 实现二分法查找 二分法查找,需要数组内是一个有序的序列 二分查找比线性查找:数组的元素数越多,效率提高的越明显 二分查找的效率表示:O(log2N) N在2的M次幂范围,那查找的次数最大就是M, log2N表示2的M次幂等于 ...

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

  1. //循环,二分查找
  2. static int binarySearch(int[] array, int data) {
  3. int start = 0;
  4. int end = array.length - 1;
  5. int mid = -1;
  6. while (start <= end) {
  7. System.out.println("查找次数");
  8. mid = (start + end) >>> 1;
  9. if (array[mid] < data) {
  10. start = mid + 1;
  11. } else if (array[mid] > data) {
  12. end = mid - 1;
  13. } else {
  14. return mid;
  15. }
  16. System.out.println("start=" + start+",end="+end+",mid="+mid);
  17. }
  18. return -1;
  19. }
复制代码
  1. //递归二分查找 初始start=0, end = array.length - 1
  2. static int binarySearch4Recursion(int[] array, int data, int start, int end) {
  3. int mid = -1;
  4. System.out.println("查找次数");
  5. if (start > end) {
  6. return mid;
  7. }
  8. mid = (start + end) >>> 1;
  9. if (array[mid] < data) {
  10. return binarySearch4Recursion(array, data, mid + 1, end);
  11. } else if (array[mid] > data) {
  12. return binarySearch4Recursion(array, data, start, mid - 1);
  13. } else {
  14. return mid;
  15. }
  16. }
复制代码

二分法插入排序

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

  1. /*
  2. * 二分(折半)插入排序
  3. * 设有一个序列a[0],a[1]...a[n];其中a[i-1]前是已经有序的,当插入时a[i]时,利用二分法搜索a[i]插入的位置
  4. */
  5. public class BinaryInsertSort {
  6. public static void main(String[] args) {
  7. int len = 10;
  8. int[] ary = new int[len];
  9. Random random = new Random();
  10. for (int j = 0; j < len; j++) {
  11. ary[j] = random.nextInt(1000);
  12. }
  13. binaryInsert(ary);
  14. /*
  15. * 复杂度分析: 最佳情况,即都已经排好序,则无需右移,此时时间复杂度为:O(n lg n) 最差情况,全部逆序,此时复杂度为O(n^2)
  16. * 无法将最差情况的复杂度提升到O(n|logn)。
  17. */
  18. // 打印数组
  19. printArray(ary);
  20. }
  21. /**
  22. * 插入排序
  23. * @param ary
  24. */
  25. private static void binaryInsert(int[] ary) {
  26. int setValueCount = 0;
  27. // 从数组第二个元素开始排序,因为第一个元素本身肯定是已经排好序的
  28. for (int j = 1; j < ary.length; j++) {// 复杂度 n
  29. // 保存当前值
  30. int key = ary[j];
  31. // ? 利用二分查找定位插入位置
  32. // int index = binarySearchAsc(ary, ary[j], 0, j - 1);// 复杂度:O(logn)
  33. // int index = binarySearchDesc(ary, ary[j], 0, j - 1);// 复杂度:O(logn)
  34. int index = binarySearchDesc2(ary, ary[j], 0, j - 1);// 复杂度:O(logn)
  35. printArray(ary);
  36. System.out.println("第" + j +"个索引上的元素要插入的位置是:" + index);
  37. // 将目标插入位置,同时右移目标位置右边的元素
  38. for (int i = j; i > index; i--) {// 复杂度,最差情况:(n-1)+(n-2)+...+n/2=O(n^2)
  39. ary[i] = ary[i - 1]; //i-1 <==> index
  40. setValueCount++;
  41. }
  42. ary[index] = key;
  43. setValueCount++;
  44. }
  45. System.out.println("n 设值次数(setValueCount)=====> " + setValueCount);
  46. }
  47. /**
  48. * 二分查找 升序 递归
  49. *
  50. * @param ary
  51. * 给定已排序的待查数组
  52. * @param target
  53. * 查找目标
  54. * @param from
  55. * 当前查找的范围起点
  56. * @param to
  57. * 当前查找的返回终点
  58. * @return 返回目标在数组中,按顺序应在的位置
  59. */
  60. private static int binarySearchAsc(int[] ary, int target, int from, int to) {
  61. int range = to - from;
  62. // 如果范围大于0,即存在两个以上的元素,则继续拆分
  63. if (range > 0) {
  64. // 选定中间位
  65. int mid = (to + from) / 2;
  66. // 如果临界位不满足,则继续二分查找
  67. if (ary[mid] > target) {
  68. /*
  69. * mid > target, 升序规则,target较小,应交换位置 前置, 即target定位在mid位置上,
  70. * 根据 查找思想, 从from到 mid-1认为有序, 所以to=mid-1
  71. */
  72. return binarySearchAsc(ary, target, from, mid - 1);
  73. } else {
  74. /*
  75. * mid < target, 升序规则,target较大,不交换位置,查找比较的起始位置应为mid+1
  76. */
  77. return binarySearchAsc(ary, target, mid + 1, to);
  78. }
  79. } else {
  80. if (ary[from] > target) {//如 5,4, 要插入的是4
  81. return from;
  82. } else {
  83. return from + 1;
  84. }
  85. }
  86. }
  87. /**
  88. * 二分查找 降序, 递归
  89. */
  90. private static int binarySearchDesc(int[] ary, int target, int from, int to) {
  91. int range = to - from;
  92. if (range > 0) {
  93. int mid = (from + to) >>> 1;
  94. if (ary[mid] > target) {
  95. return binarySearchDesc(ary, target, mid + 1, to);
  96. } else {
  97. return binarySearchDesc(ary, target, from, mid - 1);
  98. }
  99. } else {
  100. if (ary[from] > target) {//如 5,4, 要插入的是4
  101. return from + 1;
  102. } else {
  103. return from;
  104. }
  105. }
  106. }
  107. /**
  108. * 二分查找 降序, 非递归
  109. */
  110. private static int binarySearchDesc2(int[] ary, int target, int from, int to) {
  111. // while(from < to) {
  112. for (; from < to; ) {
  113. int mid = (from + to) >>> 1;
  114. if (ary[mid] > target) {
  115. from = mid + 1;
  116. } else {
  117. to = mid -1;
  118. }
  119. }
  120. //from <==> to;
  121. if (ary[from] > target) {//如 5,4, 要插入的是4
  122. return from + 1;
  123. } else {
  124. return from;
  125. }
  126. }
  127. private static void printArray(int[] ary) {
  128. for (int i : ary) {
  129. System.out.print(i + " ");
  130. }
  131. }
  132. }
复制代码

打印

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

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

  1. 931 918 706 562 531 442 333 216 210 132
复制代码

最新评论

小黑屋|在路上 ( 蜀ICP备15035742号-1 

;

GMT+8, 2025-5-6 12:31

Copyright 2015-2025 djqfx

返回顶部