最骚操作的二分查找,秀儿?
发布日期:2021-06-30 12:37:16
浏览次数:3
分类:技术文章
本文共 7226 字,大约阅读时间需要 24 分钟。
文章目录
你不知道的事
你肯定听说过在有序数组中,通过二分算法查找等于指定的值?但是…
你是否听说过在有序数组中,通过二分算法查找大于等于指定的值的最左下标?
你是否听说过在有序数组中,通过二分算法查找小于等于指定的值的最右下标?
你是否听说过在无序数组中,也可以用二分算法?知道如何用二分算法解决局部最小值问题吗?
骚算法
package com.nobody.search;/** * @author Mr.nobody * @Description 二分查找 * @date 2020/9/5 */public class Code01_BinarySearch { // 二分查找,在有序数组中查找指定值,找到返回下标,否则返回-1 public static int binarySearch(int[] sortedArr, int num) { if (null == sortedArr || sortedArr.length == 0) { return -1; } int left = 0; int right = sortedArr.length - 1; int mid = 0; while (left <= right) { // 等同于mid = (left + right) / 2; // 但是下面的表达式既速度快,还能避免left+right溢出 mid = left + ((right - left) >> 1); if (sortedArr[mid] == num) { return mid; } else if (sortedArr[mid] > num) { right = mid - 1; } else { left = mid + 1; } } return -1; } // 二分查找,在有序数组中查找大于等于指定值的最左的数,找到返回下标,否则返回-1 // 例如数组[1, 2, 2, 5, 8, 10, 11, 11, 11, 20],指定值为2,则返回下标1 // 指定值为7,则返回下标为4;指定值为0,返回下标0;指定值为25,则返回下标-1 public static int greaterEqualBinarySearch(int[] sortedArr, int num) { if (null == sortedArr || sortedArr.length == 0) { return -1; } int left = 0; int right = sortedArr.length - 1; int mid = 0; // 记录满足的最左下标 int mostLeftIndex = -1; while (left <= right) { // 等同于mid = (left + right) / 2; // 但是下面的表达式既速度快,还能避免left+right溢出 mid = left + ((right - left) >> 1); if (sortedArr[mid] >= num) { mostLeftIndex = mid; right = mid - 1; } else { left = mid + 1; } } return mostLeftIndex; } // 二分查找,在有序数组中查找小于等于指定值的最右的数,找到返回下标,否则返回-1 // 例如数组[1, 2, 2, 5, 8, 10, 11, 11, 11, 20],指定值为2,则返回下标2 // 指定值为7,则返回下标为3;指定值为0,返回下标-1;指定值为25,则返回下标9 public static int lessEqualBinarySearch(int[] sortedArr, int num) { if (null == sortedArr || sortedArr.length == 0) { return -1; } int left = 0; int right = sortedArr.length - 1; int mid = 0; // 记录满足的最右下标 int mostRightIndex = -1; while (left <= right) { // 等同于mid = (left + right) / 2; // 但是下面的表达式既速度快,还能避免left+right溢出 mid = left + ((right - left) >> 1); if (sortedArr[mid] <= num) { mostRightIndex = mid; left = mid + 1; } else { right = mid - 1; } } return mostRightIndex; } // 二分查找,在数组中(不一定要有序)查找局部最小数(随便一个就可以),找到返回下标,否则返回-1 // 何为局部最小值,当某一个位置i的数,只要小于等于i-1和i+1位置的数时,即i位置的数就是局部最小值 // 如果不存在i-1或i+1位置,就不需要比较,例如0位置的数只需要小于等于1位置的数即可; // n-1位置的数只需要小于等于n-2位置的数即可; public static int localMinBinarySearch(int[] arr) { if (null == arr || arr.length == 0) { return -1; } if (arr.length == 1 || arr[0] <= arr[1]) { return 0; } if (arr[arr.length - 1] <= arr[arr.length - 2]) { return arr.length - 1; } // 既然数组头部和尾部都不满足局部最小值,那肯定整个数据区间是↘...↗走势 // 则可以从中间切入,局部最小肯定在中间,左区间或者右区间存在, // 按此逻辑进行二分查找下去即可 int left = 0; int right = arr.length - 1; int mid = 0; while (left <= right) { // 等同于mid = (left + right) / 2; // 但是下面的表达式既速度快,还能避免left+right溢出 mid = left + ((right - left) >> 1); if (arr[mid - 1] >= arr[mid] && arr[mid] <= arr[mid + 1]) { // 既小于等于左边,又小于等于右边,满足局部最小值 return mid; } else if (arr[mid - 1] < arr[mid]) { // 大于左边,则左边是↘...↗走势,那肯定在左边区间一定存在局部最小值 right = mid - 1; } else if (arr[mid] > arr[mid + 1]) { // 大于右边,则右边是是↘...↗走势,那肯定在右边区间一定存在局部最小值 left = mid + 1; } } return -1; }}
测试
package com.nobody.search;import com.nobody.sort.Code01_InsertionSort;import java.util.Arrays;/** * @author Mr.nobody * @Description * @date 2020/9/6 */public class Main { public static void main(String[] args) { int[] arr = { 1, 2, 2, 5, 8, 10, 11, 11, 11, 20}; int num = 2; System.out.print("有序数组:"); Arrays.stream(arr).forEach(e -> System.out.print(e + " ")); System.out.println(); System.out.println("待查找数:" + num); System.out.println("查找的下标:" + Code01_BinarySearch.binarySearch(arr, num)); System.out.println("----------------------------------------"); System.out.println("大于等于的数:" + 2 + ",最左下标为:" + Code01_BinarySearch.greaterEqualBinarySearch(arr, 2)); System.out.println("大于等于的数:" + 7 + ",最左下标为:" + Code01_BinarySearch.greaterEqualBinarySearch(arr, 7)); System.out.println("大于等于的数:" + 0 + ",最左下标为:" + Code01_BinarySearch.greaterEqualBinarySearch(arr, 0)); System.out.println("大于等于的数:" + 25 + ",最左下标为:" + Code01_BinarySearch.greaterEqualBinarySearch(arr, 25)); System.out.println("----------------------------------------"); System.out.println("小于等于的数:" + 2 + ",最右下标为:" + Code01_BinarySearch.lessEqualBinarySearch(arr, 2)); System.out.println("小于等于的数:" + 7 + ",最右下标为:" + Code01_BinarySearch.lessEqualBinarySearch(arr, 7)); System.out.println("小于等于的数:" + 0 + ",最右下标为:" + Code01_BinarySearch.lessEqualBinarySearch(arr, 0)); System.out.println("小于等于的数:" + 25 + ",最右下标为:" + Code01_BinarySearch.lessEqualBinarySearch(arr, 25)); System.out.println("----------------------------------------"); int[] arr1 = { 1, 2, 2, 10, 8, 4, 2, 11, 7, 20}; System.out.print("数组:"); Arrays.stream(arr1).forEach(e -> System.out.print(e + " ")); System.out.println(",局部最小值下标:" + Code01_BinarySearch.localMinBinarySearch(arr1)); int[] arr2 = { 4, 2, 2, 10, 8, 4, 2, 11, 7, 5}; System.out.print("数组:"); Arrays.stream(arr2).forEach(e -> System.out.print(e + " ")); System.out.println(",局部最小值下标:" + Code01_BinarySearch.localMinBinarySearch(arr2)); int[] arr3 = { 7, 2, 2, 10, 8, 4, 2, 11, 7, 20}; System.out.print("数组:"); Arrays.stream(arr3).forEach(e -> System.out.print(e + " ")); System.out.println(",局部最小值下标:" + Code01_BinarySearch.localMinBinarySearch(arr3)); int[] arr4 = { 3, 2, 4, 10, 8, 6, 7, 11, 7, 10}; System.out.print("数组:"); Arrays.stream(arr4).forEach(e -> System.out.print(e + " ")); System.out.println(",局部最小值下标:" + Code01_BinarySearch.localMinBinarySearch(arr4)); }}
测试结果
有序数组:1 2 2 5 8 10 11 11 11 20 待查找数:2查找的下标:1----------------------------------------大于等于的数:2,最左下标为:1大于等于的数:7,最左下标为:4大于等于的数:0,最左下标为:0大于等于的数:25,最左下标为:-1----------------------------------------小于等于的数:2,最右下标为:2小于等于的数:7,最右下标为:3小于等于的数:0,最右下标为:-1小于等于的数:25,最右下标为:9----------------------------------------数组:1 2 2 10 8 4 2 11 7 20 ,局部最小值下标:0数组:4 2 2 10 8 4 2 11 7 5 ,局部最小值下标:9数组:7 2 2 10 8 4 2 11 7 20 ,局部最小值下标:6数组:3 2 4 10 8 6 7 11 7 10 ,局部最小值下标:5
转载地址:https://javalib.blog.csdn.net/article/details/108428294 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!
发表评论
最新留言
路过按个爪印,很不错,赞一个!
[***.219.124.196]2024年05月04日 11时03分37秒
关于作者
喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!
推荐文章
小程序图片转Base64,方法总结
2019-05-01
element中路由跳转以后激活当前菜单高亮
2019-05-01
VUE中同级页面传参的方式
2019-05-01
微信小程序setData复杂数组的更新、删除、添加、拼接
2019-05-01
has been blocked by CORS policy: Response to preflight request doesn‘t pass access control check 报错
2019-05-01
微信小程序 分享的图片使用canvas生成
2019-05-01
JS算法之累加
2019-05-01
从四川电视台播放事故论AI智能审核的重要性
2019-05-01
List集合排序Comparable与Comparator实现
2019-05-01
你为什么要写这样的代码?你知不知道填这个坑有多难
2019-05-01
查看生产环境日记常用的几个linux命令
2019-05-01
spring+quartz.2.3.0数据库持久化实现
2019-05-01
Malformed \uxxxx encoding.异常
2019-05-01
使用aspose.words 18.6实现pdf文档转换
2019-05-01
Java 生成文字图片
2019-05-01