这篇博客的二分用的都是左闭右闭的区间,对于二分来说还是我还是习惯这样写
最传统的二分查找,用左闭右闭写
  int search(vector<int>& nums, int target) {
        int left = 0;
        int right = nums.size() - 1; // 定义target在左闭右闭的区间里,[left, right]
        while (left <= right) { // 当left==right,区间[left, right]依然有效,所以用 <=
            int middle = left + ((right - left) / 2);// 防止溢出 等同于(left + right)/2
            if (nums[middle] > target) {
                right = middle - 1; // target 在左区间,所以[left, middle - 1]
            } else if (nums[middle] < target) {
                left = middle + 1; // target 在右区间,所以[middle + 1, right]
            } else { // nums[middle] == target
                return middle; // 数组中找到目标值,直接返回下标
            }
        }
        // 未找到目标值
        return -1;
    }左闭右开来写,注意的就是当中间值大于目标值的时候,right可不是mid-1,而是mid,因为右边是开的
 int search(vector<int>& nums, int target) {
        int left = 0;
        int right = nums.size(); // 定义target在左闭右开的区间里,即:[left, right)
        while (left < right) { // 因为left == right的时候,在[left, right)是无效的空间,所以使用 <
            int middle = left + ((right - left) >> 1);
            if (nums[middle] > target) {
                right = middle; // target 在左区间,在[left, middle)中
            } else if (nums[middle] < target) {
                left = middle + 1; // target 在右区间,在[middle + 1, right)中
            } else { // nums[middle] == target
                return middle; // 数组中找到目标值,直接返回下标
            }
        }
        // 未找到目标值
        return -1;
    }用二分去找目标值的左边界和右边界的问题,当然我们可以去先普通二分找到目标值,再在这个目标值的位置开始左右去线性探测找左右边界,但是当目标值重复太多,你这么写二分就不是二分了,不就是On线性去找
所以这里写的都是纯二分的思路,还是用左闭右闭去写,好好体会代码逻辑
左边界里我多定义一个变量去表示,整体代码和右边界可能会有一丢丢出入(右边界的看着简单),但是俩逻辑都是一模一样,就是写的方式不同,都看懂了才能明白这个怎么去找边界
先放两张图去体会一下,什么是二分找边界。这题是力扣34. 在排序数组中查找元素的第一个和最后一个位置,我分开写去分别表示左边界和右边界
 
 
找左边界的模板代码
 int getLeftBorder(vector<int>& nums, int target) {
        int left = 0;
        int right = nums.size() - 1;
        int leftBorder = -2; // 记录一下leftBorder没有被赋值的情况
        while (left <= right) {
            int middle = left + ((right - left) / 2);
            if (nums[middle] >target) { // 寻找左边界,nums[middle] == target的时候更新right
                right = middle - 1;
               
            } else if(nums[middle] <target){
                left = middle + 1;
            }
            else if(nums[middle] ==target)
            {         
                leftBorder = middle;  
                right = middle - 1;
            }
        }
        return leftBorder;
    }找右边界的模板代码,俩其实都能改成一样的
  int getRightBorder(vector<int>& nums, int target) {
        int left = 0;
        int right = nums.size() - 1;
 
        while (left <= right) {
            int middle = left + ((right - left) / 2);
            if (nums[middle] > target)
            {
                right = middle - 1;
            } 
            else if(nums[middle] <= target)
            { // 寻找右边界,nums[middle] == target的时候更新left
                        
                left = middle + 1;  
            }               
        }
        return right;
    }今天的记录就到这了..2022/11/10
希望努力会有回报



















