• 数的范围---二分法


    题目描述

    给定一个按照升序排列的长度为n的整数数组,以及 q 个查询。对于每个查询,返回一个元素k的起始位置和终止位置(位置从0开始计数)。如果数组中不存在该元素,则返回“-1 -1”。

    例如

    num = [1,2,2,3,3,4];

    • 查找3

            返回结果:3, 4

    • 查找4

            返回结果:5, 5

    • 查找5

            返回结果:-1, -1

    这里以 查找3 为例进行讲解。

    假设此时的情景为,需要我们找到第一个>=3的数字,试想一下,如果能把整个区间分了两半,左边是<3的区间,右边是>=3的区间 如图:

    那么右区间的第一个点,就是我们找到的符合>=3的第一个数字,将区间分为两半,是不是非常清晰!

    1. int l = -1;
    2. int r = q.length;
    3. while (l + 1 != r) {
    4. int mid = (l + r) >> 1;
    5. if (q[mid] >= x) {
    6. r = mid;
    7. } else {
    8. l = mid;
    9. }
    10. // 此时r 就是第一个3出现的位置
    11. }

    【同理】第二个3出现的位置为

    1. l = -1;
    2. r = q.length ;
    3. while (l + 1 != r) {
    4. int mid = (l + r ) >> 1;
    5. if (q[mid] <= x) {
    6. l = mid;
    7. } else {
    8. r = mid;
    9. }
    10. }
    11. System.out.println(l);

    【代码】最后给出两种实现--但是推荐采用binaryModel函数

    1. package algorithm;
    2. import java.util.Scanner;
    3. public class BinaryModel {
    4. public static void main(String[] args) {
    5. int[] q = {1, 2, 2, 3, 3, 4};
    6. binarySort(q, 5);
    7. }
    8. public static void binarySort(int[] q, int x) {
    9. int l = 0;
    10. int r = q.length - 1;
    11. while (l < r) {
    12. int mid = (l + r) >> 1;
    13. if (q[mid] >= x) {
    14. r = mid;
    15. } else {
    16. l = mid + 1;
    17. }
    18. }
    19. if (q[r] != x) {
    20. System.out.println("-1 -1");
    21. } else {
    22. System.out.print(r + " ");
    23. l = 0;
    24. r = q.length - 1;
    25. while (l < r) {
    26. int mid = (l + r + 1) >> 1;
    27. if (q[mid] <= x) {
    28. l = mid;
    29. } else {
    30. r = mid - 1;
    31. }
    32. }
    33. System.out.println(l);
    34. }
    35. }
    36. public static void binaryModel(int[] q, int x) {
    37. int l = -1;
    38. int r = q.length;
    39. while (l + 1 != r) {
    40. int mid = (l + r) >> 1;
    41. if (q[mid] >= x) {
    42. r = mid;
    43. } else {
    44. l = mid;
    45. }
    46. // 此时r 就是第一个3出现的位置
    47. }
    48. if (q[r] != x) {
    49. System.out.println("-1 -1");
    50. } else {
    51. System.out.print(r + " ");
    52. l = -1;
    53. r = q.length ;
    54. while (l + 1 != r) {
    55. int mid = (l + r ) >> 1;
    56. if (q[mid] <= x) {
    57. l = mid;
    58. } else {
    59. r = mid;
    60. }
    61. }
    62. System.out.println(l);
    63. }
    64. }
    65. }

     

  • 相关阅读:
    关于python字符串
    UI设计公司成长日记2:修身及持之以恒不断学习是要务
    第1关:Hive的安装与配置
    MySQL导入.sql文件方法以及导入失败的问题解决
    智能运维应用之道,告别企业数字化转型危机
    unity 随机生成物体方法
    Oracle数据库基础
    容猫科技PHP面试题(!带答案)
    Vue(四)——全局事件总线, 消息订阅与发布 ,nextTick
    【Java面试】工作7年去字节面试竟然在这题翻车了,请你说一下你对时间轮的理解?
  • 原文地址:https://blog.csdn.net/abc123mma/article/details/127663753