二分查找

------

二分查找基本版及其各种变形的汇总

思想:  二分搜索的核心就是**循环结束条件**和**左右边界迭代规则**

#### 一. 基本二分

##### 基本的二分查找

我们这个算法中使用的是前者 `[left, right]` 两端都闭的区间。**这个区间其实就是每次进行搜索的区间**

当然,找到了目标值的时候可以终止

if(nums[mid] == target)

​        return mid;

```java

int binarySearch(int[] nums, int target) {

    int left = 0, right = nums.length - 1;

    while(left <= right) {

        int mid = left + (right - left) / 2;

        if (nums[mid] < target) {

            left = mid + 1;

        } else if (nums[mid] > target) {

            right = mid - 1;

        } else if(nums[mid] == target) {

            // 直接返回

            return mid;

        }

    }

    // 直接返回

    return -1;

}

```

#### 二. 进阶二分

##### 2.1 寻找左侧边界的二分查找

```java

int leftBound(int[] nums, int target) {

    int left = 0, right = nums.length - 1;

    while (left <= right) {

        int mid = left + (right - left) / 2;

        if (nums[mid] < target) {

            left = mid + 1;

        } else if (nums[mid] > target) {

            right = mid - 1;

        } else if (nums[mid] == target) {

            // 别返回,锁定左侧边界

            right = mid - 1;

        }

    }

    // 最后要检查 left 越界的情况

    if (left >= nums.length || nums[left] != target)

        return -1;

    return left;

}

```

##### 2.2 寻找右侧边界的二分查找

```java

int rightBound(int[] nums, int target) {

    int left = 0, right = nums.length - 1;

    while (left <= right) {

        int mid = left + (right - left) / 2;

        if (nums[mid] < target) {

            left = mid + 1;

        } else if (nums[mid] > target) {

            right = mid - 1;

        } else if (nums[mid] == target) {

            // 别返回,锁定右侧边界

            left = mid + 1;

        }

    }

    // 最后要检查 right 越界的情况

    if (right < 0 || nums[right] != target)

        return -1;

    return right;

}

```

注意点:**注意搜索区间 和 while的终止条件**

##### 2.3 查找最后一个等于或者小于target的元素

关键词:**右边界**

查找最后一个等于或者小于target的元素,也就是说如果查找target值的元素有好多个,返回这些元素最右边的元素下标;如果没有等于target值的元素,则返回小于target的最右边元素下标

```java

// 查找最后一个等于或者小于target的元素

int findLastEqualSmaller(int[] nums, int target) {

    int left = 0;

    int right = nums.length - 1;

    // 这里必须是 <=

    while (left <= right) {

        int mid = (left + right) / 2;

        if (nums[mid] > target) {

            right = mid - 1;

        }

        else {

            left = mid + 1;

        }

    }

    return right;  //必须返回right

}

```

##### 2.4 查找最后一个小于target的元素

返回小于target的最右边元素下标

```java

// 查找最后一个小于target的元素

int findLastSmaller(int[] nums, int target) {

    int left = 0;

    int right = nums.length - 1;

    // 这里必须是 <=

    while (left <= right) {

        int mid = (left + right) / 2;

        if (nums[mid] >= target) {

            right = mid - 1;

        }

        else {

            left = mid + 1;

        }

    }

    return right;  //必须返回right

}

```

##### 2.5 查找第一个等于或者大于target的元素

  关键词:左边界

 查找第一个等于或者大于target 的元素,也就是说如果查找target值的元素有好多个,返回这些元素最左边的元素下标;如果没有等于target值的元素,则返回大于target的最左边元素下标

```java

// 查找第一个等于或者大于key的元素

int findFirstEqualLarger(int[] nums, int target) {

    int left = 0;

    int right = nums.length - 1;

    // 这里必须是 <=

    while (left <= right) {

        int mid = (left + right) / 2;

        if (nums[mid] >= target) {

            right = mid - 1;

        }

        else {

            left = mid + 1;

        }

    }

    return left;

}

```

##### 2.6 查找第一个大于target的元素

```java

// 查找第一个大于key的元素

int findFirstLarger(int[] nums, int target) {

    int left = 0;

    int right = nums.length - 1;

    // 这里必须是 <=

    while (left <= right) {

        int mid = (left + right) / 2;

        if (nums[mid] > target) {

            right = mid - 1;

        }

        else {

            left = mid + 1;

        }

    }

    return left;

}

```

第一个大于等于target 或者  最后一个小于等于target的元素,扩展开来,

最小条件:  **第一个满足条件的** ,左边界

**案例一:阿珂吃香蕉**:找到第一个满足条件的,比 该值大的都满足,所以是找左边界

满足条件指的是:canFinish(piles, mid, H)

https://leetcode-cn.com/problems/koko-eating-bananas/

```java

public int minEatingSpeed(int[] piles, int H) {

        int maxSpeed = getMax(piles);

        int left= 1;

        int right = maxSpeed;

        while(left <= right){

            int mid= left +(right -left)/2;

            if(canFinish(piles, mid, H)){  //查找左边界,找到第一个满足 条件的

                right = mid-1;

            } else {

                left = mid + 1;

            }

        }

        return left;

    }

    // 时间复杂度 O(N)

    boolean canFinish(int[] piles, int speed, int H) {

        int time = 0;

        for (int n : piles) {

            time += timeOf(n, speed);

        }

        return time <= H;

    }

    int timeOf(int n, int speed) {

        return (n / speed) + ((n % speed > 0) ? 1 : 0);

    }

    int getMax(int[] piles) {

        int max = 0;

        for (int n : piles)

            max = Math.max(n, max);

        return max;

    }

```

案例二: 给定一个包含 n + 1 个整数的数组 nums, 找出重复数

满足条件: cnt[i] > i;

```java

    public int findDuplicate_BinarySearch(int[] nums) {

        int n  = nums.length;

        int l =1, r = n-1;

        while(l <=r ){

          int mid = l + (r -l)/2;

            int cnt = cntLessEqual(nums, mid);

            if(cnt > mid){//  查找第一个满足条件的 :cnt[mid] > mid的最左边

                r = mid -1;

            } else {

                l = mid +1;

            }

        }

        return l;

    }

    private int cntLessEqual(int[] nums, int target){

        int cnt = 0;

        for(int n:nums){

            if(n <= target){

                cnt++;

            }

        }

        return cnt;

    }

```

#### 三. 二分查找变种总结

```java

// 这里必须是 <=

while (left <= right) {

    int mid = (left + right) / 2;

    if (array[mid] ? key) {

        //... right = mid - 1;

    }

    else {

        // ... left = mid + 1;

    }

}

return xxx;

```

1. 首先判断出是返回left,还是返回right

  因为我们知道最后跳出while (left <= right)循环条件是right < left,且right = left - 1。最后right和left一定是卡在"边界值"的左右两边,如果是比较值为key,查找小于等于(或者是小于)key的元素,则边界值就是等于key的所有元素的最左边那个,其实应该返回left。

    以数组{1, 2, 3, 3, 4, 5}为例,如果需要查找第一个等于或者小于3的元素下标,我们比较的key值是3,则最后left和right需要满足以下条件:

  ![img](http://images2015.cnblogs.com/blog/772134/201608/772134-20160813153101328-934983178.png)

    我们比较的key值是3,所以此时我们需要返回left

2. 判断出比较符号

  ```java

  int mid = (left + right) / 2;

  if (array[mid] ? key) {

      //... right = xxx;

  }

  else {

      // ... left = xxx;

  }

  ```

   也就是这里的 if (array[mid] ? key) 中的判断符号,结合步骤1和给出的条件,如果是查找小于等于key的元素,则知道应该使用判断符号>=,因为是要返回left,所以如果array[mid]等于或者大于key,就应该使用>=,以下是完整代码

  ```java

  // 查找小于等于key的元素

  int mid = (left + right) / 2;

  if (array[mid] >= key) {

      right = mid - 1;

  }

  else {

      left = mid + 1;

  }

  ```

### 另类二叉查找

https://leetcode-cn.com/problems/find-peak-element/

寻找峰值

![image.png](https://pic.leetcode-cn.com/802bad70c4444bf708f4c63e30e054a33c27ace43b3c7b4fa64a0ffb8201fb7d-image.png)

峰值元素是指其值大于左右相邻值的元素。

给定一个输入数组 nums,其中 nums[i] ≠ nums[i+1],找到峰值元素并返回其索引。

数组可能包含多个峰值,在这种情况下,返回任何一个峰值所在位置即可。

你可以假设 nums[-1] = nums[n] = -∞

```java

public int findPeakElement(int[] nums) {


        int l = 0, r = nums.length - 1;

        while (l < r) {

            int mid =  l + (r -l)/2;

            if (nums[mid] > nums[mid + 1])

                r = mid;

            else

                l = mid + 1;

        }

        return l;

    }

```

参考:

https://labuladong.gitbook.io/algo/suan-fa-si-wei-xi-lie/er-fen-cha-zhao-xiang-jie

https://blog.csdn.net/bat67/article/details/72049104

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351