当前位置:   article > 正文

【算法】递归实现二分查找(优化)以及非递归实现二分查找

【算法】递归实现二分查找(优化)以及非递归实现二分查找
递归实现二分查找 
思路分析

1.首先确定该数组中间的下标 mid = (left + right) / 2;

2.然后让需要查找的数 findVal 和 arr[mid] 比较

  1. findVal > arr[mid],说明要查找的数在 arr[mid] 右边,需要向右递归
  2. findVal < arr[mid],说明要查找的数在 arr[mid] 左边,需要向左递归
  3. findVal == arr[mid],说明找到,返回

什么时候结束递归

1.找到就结束递归

2.递归完整个数组,依然没有找到,即 left > right 就阶结束递归

  1. //注意:使用二分查找的前提是该数组是有序的
  2. public class BinarySearch {
  3. public static void main(String[] args) {
  4. int[] arr = {-3, 1, 18, 99, 1000, 1024};
  5. int findVal = 99;
  6. int index = binarySearch(arr, 0, arr.length - 1, findVal);
  7. System.out.printf("%d的索引为%d\n", findVal, index);
  8. }
  9. //二分查找算法
  10. public static int binarySearch(int[] arr, int left, int right, int findVal) {
  11. //如果找不到,返回 -1
  12. if (left > right) {
  13. return -1;
  14. }
  15. int mid = (left + right) / 2;
  16. int midVal = arr[mid];
  17. if (findVal > midVal) {
  18. return binarySearch(arr, mid + 1, right, findVal);
  19. }else if (findVal < midVal) {
  20. return binarySearch(arr, left, mid - 1, findVal);
  21. } else {
  22. return mid;
  23. }
  24. }
  25. }
优化

当一个有序数组中有多个相同数值时,如{-3, 1, 18, 99,99, 1000, 1024},如何将所有数值都查找到,比如这里的 99

思路分析

1.在找到 mid 值,不要马上返回

2.向 mid 索引值的左边扫描,将所有满足的下标加入集合 ArrayList

3.向 mid 索引值的右边扫描,将所有满足的下标加入集合 ArrayList

4.将 ArrayList 返回

  1. //注意:使用二分查找的前提是该数组是有序的
  2. public class BinarySearch {
  3. public static void main(String[] args) {
  4. int[] arr = {-3, 1, 18, 99, 99, 99, 1000, 1024};
  5. int findVal = 99;
  6. ArrayList<Integer> resIndexList = binarySearch(arr, 0, arr.length - 1, findVal);
  7. System.out.printf("resIndexList = " + resIndexList);
  8. }
  9. //优化二分查找算法
  10. public static ArrayList<Integer> binarySearch(int[] arr, int left, int right, int findVal) {
  11. //如果找不到,返回 -1
  12. if (left > right) {
  13. return new ArrayList<Integer>();
  14. }
  15. int mid = (left + right) / 2;
  16. int midVal = arr[mid];
  17. if (findVal > midVal) {
  18. return binarySearch(arr, mid + 1, right, findVal);
  19. }else if (findVal < midVal) {
  20. return binarySearch(arr, left, mid - 1, findVal);
  21. } else {
  22. /*
  23. 1.在找到 mid 值,不要马上返回
  24. 2.向 mid 索引值的左边扫描,将所有满足的下标加入集合 ArrayList
  25. 3.向 mid 索引值的右边扫描,将所有满足的下标加入集合 ArrayList
  26. 4.将 ArrayList 返回
  27. */
  28. ArrayList<Integer> resIndexlist = new ArrayList<>();
  29. //向左扫描
  30. int temp = mid - 1;
  31. while (true) {
  32. if (temp < 0 || arr[temp] != findVal) { //退出
  33. break;
  34. }
  35. //否则,就将 temp 放入到 resIndexlist
  36. resIndexlist.add(temp);
  37. temp -= 1; //temp 左移
  38. }
  39. resIndexlist.add(mid);
  40. //向右扫描
  41. temp = mid + 1;
  42. while (true) {
  43. if (temp > arr.length - 1 || arr[temp] != findVal) { //退出
  44. break;
  45. }
  46. //否则,就将 temp 放入到 resIndexlist
  47. resIndexlist.add(temp);
  48. temp += 1; //temp 左移
  49. }
  50. return resIndexlist;
  51. }
  52. }
  53. }

非递归实现二分查找
  1. public class BinarySearchNoRecursion {
  2. public static void main(String[] args) {
  3. int[] arr = {1, 2, 8, 999, 1024, 1234};
  4. int target = 8;
  5. System.out.println("待查询数组为:" + Arrays.toString(arr));
  6. System.out.printf("%d的下标索引为%d", target, binarySearch(arr, target));
  7. }
  8. /**
  9. *
  10. * @param arr 待查找的数组,升序
  11. * @param target 待查找的数
  12. * @return 返回对应的下标,-1 表示没有找到
  13. */
  14. public static int binarySearch(int[] arr, int target) {
  15. int left = 0;
  16. int right = arr.length - 1;
  17. while (left <= right) { //说明可以继续查找
  18. int mid = (left + right) / 2;
  19. if (arr[mid] == target) {
  20. return mid;
  21. } else if (arr[mid] > target) {
  22. right = mid - 1;
  23. } else {
  24. left = mid + 1;
  25. }
  26. }
  27. return -1;
  28. }
  29. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/932941
推荐阅读
相关标签
  

闽ICP备14008679号