算法训练-二分查找

34. 在排序数组中查找元素的第一个和最后一个位置

在这里插入图片描述
题目链接

vector<int> searchRange(vector<int>& nums, int target) {
	     int left=0,right=nums.size()-1;
	     vector<int> ret{-1,-1};
	     while(left<=right)//区间不为空,则可以继续去找
	     {
	         int mid=(left+right)/2;
	         //int mid=left+(right-left)/2;这样写不会越界,如果left和right都很大,相加就可能越界
	         if(nums[mid]==target)
	         {
	             int i;
	             for(i=mid-1;i>=0&&nums[i]>=target;i--)
	             /*找到target不能立即存放到ret中,因为题目要求找到第一个和最后一个target位置,先从mid往前找,相同 就给ret[0]赋当前下标,对mid右变处理相同
	             因为是有序的,所以加上nums[i]>=target,往前找,不过遇到大于target则前面不可能再有了
	             */

	             if(nums[i]==target)
	             ret[0]=i;
	             if(ret[0]==-1)
	             ret[0]=mid;

	             for(i=mid+1;i<nums.size()&&nums[i]<=target;i++)
	             if(nums[i]==target)
	             ret[1]=i;
	             if(ret[1]==-1)
	             ret[1]=mid;
	         }
	         if(nums[mid]<target)
	         left=mid+1;
	         else right=mid-1;
	     }
	     return ret;

似乎这样写没有问题,并且也通过了测试用例,但是仔细分析就会发现,在某些特定数组的情况
比如:target=2 序列是:1 22222222222222222222222222222222…2222 1
中间有很多很多2,那么这个算法的效率就不是O(logn)了,接近O(n)

正确写法:

     那么问题出在哪里,因为我们的二分是对小于(mid<target)进行mid+1,大于进行mid-1,相等就往前找,和往后找,这里简单理解为return
     而应该在相等时也不return,相等应该归类于大于的情况,说明前面还可能有,继续找
     这时就可以理解为right指针不断向前,并且是跳跃式向前找,target,最后left的位置就是第一个target的位置,
     为什么是left而不是right,因为right不断向前最中总能找到一个不等的或者区间为空了

     1.不等情况:此时right指向的不等于target,那么肯定是mid小于了,left指针此时向后移动,如此循环直到区间只剩一个元素,则该元素就是第一个target
     2.区间为空:则不存在这样的元素,那么此时left和right的情况是怎样的?
     区间为空说明mid一次都没有匹配target,则left一直后移,最终left指向right+1,全过程right都没有变化
     区间为空还有一种情况是数组中没有target,但是两遍是符合的,也就是left和right都会移动
    inline int lower_bound(vector<int>& nums,int target)
    {
        int left=0,right=nums.size()-1;
        while(left<=right)
        {
            int mid=left+(right-left)/2;
            if(nums[mid]<target)
            left=mid+1;
            else 
            right=mid-1;//为什么没有相等,以及下面为什么返回left,都在上面解释过了
        }
        return left;
    }
    vector<int> searchRange(vector<int>& nums, int target) {
        int  start=lower_bound(nums,target);
        if(start==nums.size()||nums[start]!=target)//区间为空有两种情况
        return {-1,-1};
        int end=lower_bound(nums,target+1)-1;//end要找target+1,最后在-1找前一个
        return {start,end};
    }
};

返回值为vector<int>为什么可以返回return {start,end};
因为 vector<int> arr{ 1, 1};可以用初始化列表初始化容器vector

感觉第二种不好理解啊,low_bound还行,但是在searchRange中比较复杂,还是第一种的思路简单,有没有什么办法优化第一种呢?

方法一的问题在于,遇到特殊相等之后就不在是二分算法了,那么应该当相等时,继续二分,更新边界,那边界应该如何去选择呢,这就要看是要找第一个target还是第二个target,对于ret[0],则要更新right=mid+1
对于ret[1],更新left=mid-1,分两次二分就可以解决

class Solution {
public:
    vector<int> searchRange(vector<int>& nums, int target) {
        return {searchLeftOrRightBound(nums, target, "left"), searchLeftOrRightBound(nums, target, "right")};
    }
private:
    int searchLeftOrRightBound(vector<int>& nums, int target, const string& bound) {
        int left = 0, right = nums.size() - 1;
        int res = -1;
        while (left <= right) {
            int mid = (left + right) >> 1;
            if (nums[mid] < target) {
                left = mid + 1;
            }
            else if (nums[mid] > target) {
                right = mid - 1;
            }
            else {
                res = mid;
                if (bound == "left") {
                    right = mid - 1;
                }
                else if (bound == "right") {
                    left = mid + 1;
                }
            }
        }
        return res;
    }
};

162. 寻找峰值

在这里插入图片描述
题目链接

class Solution {
public:
/*这道题,最最最重要的是条件,条件,条件,两边都是负无穷,数组当中可能有很多波峰,也可能只有一个,如果尝试画图,就跟股票信息一样,没有规律,如果根据中点値判断我们的二分方向该往何处取, 这道题还有只是返回一个波峰。你这样想,中点所在地方,可能是某座山的山峰,山的下坡处,山的上坡处,如果是山峰,最后会二分终止也会找到,关键是我们的二分方向,并不知道山峰在我们左边还是右边,送你两个字你就明白了,爬山(没错,就是带你去爬山),如果你往下坡方向走,也许可能遇到新的山峰,但是也许是一个一直下降的坡,最后到边界。但是如果你往上坡方向走,就算最后一直上的边界,由于最边界是负无穷,所以就一定能找到山峰,总的一句话,往递增的方向上,二分,一定能找到,往递减的方向只是可能找到,也许没有。*/
//二分的过程就是模拟爬山,寻找波峰
    int findPeakElement(vector<int>& nums) {
        int left=0,right=nums.size()-1;

        while(left<right)
        {
            int mid=(left+right)/2;
            if(nums[mid]<nums[mid+1])//左侧小于右侧,说明是在爬坡,并且题目说nums[n] = -oo
            left=mid+1;
            else right=mid;//不能是mid-1,因为mid也有可能是个波峰
        }
        return left;
    }
};

153. 寻找旋转排序数组中的最小值

在这里插入图片描述
题目链接

class Solution {
    public:
    /*
    根据题意可以知道:该序列有两段上坡,可以视为一段高坡,一段低坡
    并且对于末尾元素x,最小值右侧都大于x,最小值左侧都小于x
    */
    int findMin(vector<int> & nums) {
        int left = 0, right = nums.size() - 1;
        while (left < right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] < nums[nums.size() - 1]) //和末尾元素比较。mid小于末尾,说明在低坡。最小值一定在mid左侧,并且mid的位置有可能就是最小值位置,所以是right=mid(这是因为最小值一定在小坡上,小坡坡底)
                right = mid;
            else //mid 大于末尾,说明在一定在大坡,大坡是不可能有最小值的,直接mid+1
                left = mid + 1;
        }
        return nums[left];
    }
};

33. 搜索旋转排序数组

在这里插入图片描述
题目链接

思路

先找到要二分的区间位置,有两个坡,那就先找最小值位置,通过比较target和最小值,来判断在哪个坡

class Solution {
    public:
    int search(vector<int> & nums, int target) {

        int left = 0, right = nums.size() - 1;
        bool asc = true;//升序标记

        if (nums[left] > nums[right])
            asc = false;

        if (asc == false) {//逆序,说明进行了旋转,有两个坡,先找坡底最小值
            if (target < nums[left] && target > nums[right])
                return -1;
            while (left < right) {
                int mid = left + (right - left) / 2;
                if (nums[mid] < nums[nums.size() - 1])
                    right = mid;
                else left = mid + 1;
            }
            if (target < nums[0])
                right = nums.size() - 1;
            else left = 0;
            cout << "left=" << left << " " << right << endl;
        }
        //确定好在哪个坡了,标准二分
        while (left <= right) {
            int mid = (left + right) / 2;
            if (nums[mid] == target)
                return mid;
            else if (nums[mid] < target)
                left = mid + 1;
            else right = mid - 1;
        }
        return -1;
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值