LeetCode 33 Search in Rotated Sorted Array 题解

来源:互联网 发布:东欧社会主义国家 知乎 编辑:程序博客网 时间:2024/06/03 11:41

题意简述:给定一个按升序排序的数组,这个数组进行了数次旋转(例如数组[0,1,2,4,5,6,7]在2和4之间执行一次旋转就会变成[4,5,6,7,0,1,2]。需要搜索某个数是否在这个数组里面。数组里面没有重复元素。
输入:数组nums,要搜索的值target
输出:如果target在nums里面,那么返回它在数组里面的位置下标;否则返回-1。
示例:对于nums=[4,5,6,7,0,1,2],target=5,应该返回5在nums中的下标1。


题解:
首先无论进行多少次旋转,数组最后只可能是两种状态:

  1. 跟原升序数组一致。
  2. 数组有一个断点,被划分为两段升序。

这是因为进行当前的旋转时,如果选择的断点跟上一次不一样,因上一次旋转而产生的两段升序又会连在一起变成一段升序。举个例子:对于数组[1,2,3,4,5],在2、3之间旋转会变成[3,4,5,1,2],断开的2和3在数组的首尾,下次在4、5之间旋转时,首尾的2和3又连在一起了([5,1,2,3,4])。如果选择的断点跟上一次一样,那就相当于撤销了上一次的旋转。

既然只有两种情况,那么我们只需加入分类讨论,就可以像排序数组一样使用二分搜索。存在断点的情况下,数更大的一段必定在数更小的一段的前面,所以考虑首尾两个值以及中间的值nums[low]、nums[up]、nums[mid](mid=(low+up)/2):

  1. 如果nums[low] < nums[mid],则数组的前半段就是升序的,直接在该段调用二分搜索;
  2. 如果nums[low] > nums[mid],则数组的前半段存在断点,无法直接调用二分搜索,因此递归调用函数在这半段继续以上的判断。
  3. 对数组的后半段的讨论跟以上两点类似。
  4. 只有在前半和后半都找不到target时才返回-1。

算法实现如下:

class Solution {private:    int binarySearch(vector<int>& nums,int low, int up, int target) {        int mid, tempup = up;        low--;up++;        while(low + 1 != up) {            mid = low + (up - low) / 2;            if(nums[mid] < target) low = mid;            else up = mid;        }        if(up > tempup || nums[up] != target) return -1;        else return up;    }    int recur(vector<int>& nums,int low, int up, int target) {        if(low == up) {            if(nums[low] == target) return low;            else return -1;        }         int mid = low + (up - low) / 2;        int resl, resr;        if(nums[low] < nums[mid]) resl = binarySearch(nums, low, mid, target);        else resl = recur(nums, low, mid, target);        if(nums[up] > nums[mid]) resr = binarySearch(nums, mid+1, up, target);        else resr = recur(nums, mid+1, up, target);        if(resl != -1) return resl;        else return resr;    }public:    int search(vector<int>& nums, int target) {        if(nums.size() == 0) return -1;        return recur(nums, 0, nums.size()-1, target);    }};
0 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 邮箱里超大附件的文件过期了怎么办 ps时间轴储存出现参数错误怎么办 苹果手机网易云会员支付失败怎么办 鲸钱包网络异常需重新加载怎么办 微信加载联系人失败点击重试怎么办 网易将军令没有帐号怎么绑定怎么办 网易将军令绑定银行卡锁定了怎么办 网易新闻上的视频没有声音怎么办 把登录窗体的宏删掉了怎么办 电脑死机了怎么办也不能关机了 虎牙直播手游无法和队友开麦怎么办 为成年开无手续麽托出车祸怎么办 支付宝商户未签约或合约过期怎么办 我的手机帐号被锁定了怎么办 点错网页下载收发收费信息怎么办 用邮箱注册的微信忘记密码了怎么办 很久不用的邮箱号忘记密码怎么办 陕西省志愿云忘记密码和邮箱怎么办 国寿易学堂忘记登录密码怎么办 百度云下载显示本地空间不足怎么办 百度云离线内容违规无法下载怎么办 百度云压缩包要压解密码没有怎么办 用2016酷我音乐制作铃音怎么办 邮箱跟手机号一同绑定支付宝怎么办 网易邮箱取消账户关联没反应怎么办 剑三重置版客户端修复不成功怎么办 网易闪电邮的邮件满了怎么办 忘了手机的账号和密码怎么办 手机绑定的扣扣达到十个上限怎么办 邮箱发送邮件到上限无法发送怎么办 路由器重置后不知道账号密码怎么办 发出的邮件别人看了撤回不了怎么办 域名和邮箱是不同的公司怎么办 qq音乐听过的歌单找不到了怎么办 快手开直播前置摄像头太暗怎么办 小米手机帐号绑定无法侠用怎么办 用交易猫买完游戏账号被骗了怎么办 一个华为账号有两个游戏账号怎么办 加密u盘电脑上打开空怎么办 国网加密u盘电脑打开为空怎么办 足球竞彩软件下架后里面的钱怎么办