当前位置: 首页 > news >正文

旋转数组中查找最小值-剑指Offer11

1.题目简介

求一个旋转数组的最小值。( 把一个数组从最开始的若干个元素搬到数组的末尾,即为旋转数组。)

  • 输入:一个递增排序数组的旋转
  • 输出:数组的最小值
  • 例子:数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1.

2.思路分析

最直观的解法是从头到尾顺序遍历,这种方法的时间复杂度为O(n)。这里并没有用到旋转数组的知识,显然不合题意。

结合提议,旋转数组从有序数组中的得来的,经过旋转后得到的两个部分也为有序的。在有序数组中查找首选二分法,可以把时间复杂度变为O(n)。由于是把递增数组中最开始的若干个元素移到末尾,那么经过旋转后,数组开头的元素一定是大于等于末尾的元素,这是循环的条件。

和二分法一样设置三个指针,low、mid和high,分别指向数组中的开头、中间和结尾元素。如果开头元素小于等于中间元素,证明前半部分是递增的,要找的最小值在后半部分,此时把low指向mid;同样的,如果中间元素的指小于等于最后元素的值,说明后面是递增序列,要找的元素在前半部分,把high指向mid。

那什么时候找到呢?当Low和high相邻时,且high所指的元素就时最小值。

例子:
arr = {3,4,5,1,2} ,low=0,high=4;mid=2;
经过依次查找,arr[low]<arr[mid];证明前半部分时递增序列,要查找的值在后半部分。
所以令low=mid=2;mid=3,第二次查找,arr[mid]=1,arr[high]=2,arr[mid]<arr[high],要查找的在后半部分
所以令high = mid = 3;
由于low和high已经相邻,找到元素,跳出循环

注意:还要考虑两种特殊情况

3.代码

    /*用二分法,查找返回旋转数组的最小值,时间复杂度为O(n)*/
    public static int getMinOfRoateArray(int[] arr,int len) throws Exception {
        int min=0;
        int low = 0;
        int high = len - 1;
        int mid = low;
        if(arr==null||len==0) {
            throw new Exception("the array is illegal");
        }       
        while(arr[low]>=arr[high]) {
            if(high-low==1) { //相邻时候找到
                mid = high;
                break;
            }           
            mid = (low+high)/2;
            //特殊情况,中间 左右的值相等,只能顺序查找了
            if(arr[low]==arr[high] && arr[mid] == arr[low]) {
                min = arr[0];
                for(int i = 1 ; i < len ; i++) {                    
                    if(arr[i]<min) {
                        min = arr[i];
                    }
                }
                return min;
            }
            //二分法
            if(arr[low]<=arr[mid]) {
                low = mid;
            }else if(arr[mid]<=arr[high]) {
                high = mid;
            }
        }
        min = arr[mid];
        return min;
    }

4.相关-二分查找

//array为要排序的整型数组
/**非递归*/
    public int binarysearch(int x) {
        if(array.isEmpty()) {
            return -1;
        }
        int low = 0;
        int high = array.size()-1;
        
        while(low <= high) {
            int mid =(int) (high+low)/2;
            if(x == array.get(mid)){
                return mid+1;
            }
            else if(x <array.get(mid)) {
                high = mid-1;
            }
            else if(x >array.get(mid)){
                low = mid+1;                
            }       
        }
        return -1;      
    }

    /**递归调用
     * */
    public int binarySearchR(int x , int low ,int high) {
        int mid = (low+high)/2;
        if(x < array.get(low)||x > array.get(high-1) ||low >high) {
            return -1;
        }
        if(x < array.get(mid)) {
            return binarySearchR(x,low, mid-1);
        }
        else if (x > array.get(mid)) {
            return binarySearchR(x,mid+1,high);
        }
        else {
            return mid+1;
        }
    }

转载于:https://www.cnblogs.com/java-learner/p/9583705.html

相关文章:

  • RIP路由信息协议
  • 服务器连接工具 secureCRT
  • SeaweedFS---01
  • Spring Boot 最佳实践(五)Spring Data JPA 操作 MySQL 8
  • 周六相约橘子洲头,共话AWS上的AI和大数据技术
  • 【转】20-TCP 协议(滑动窗口——基础)
  • httpd之apache服务器配置
  • 如何重置migration
  • LVS 之 管理工具ipvsadm介绍
  • JDBC 结构
  • 【以太坊】雷电网络的101网络原理概述
  • @property @synthesize @dynamic 及相关属性作用探究
  • 获取网贷之家数据
  • ES6 新特性之 let, const : JavaScript在变量方面的改进。
  • sqlmap tamter
  • 【跃迁之路】【733天】程序员高效学习方法论探索系列(实验阶段490-2019.2.23)...
  • 2018一半小结一波
  • Babel配置的不完全指南
  • JWT究竟是什么呢?
  • leetcode46 Permutation 排列组合
  • open-falcon 开发笔记(一):从零开始搭建虚拟服务器和监测环境
  • UMLCHINA 首席专家潘加宇鼎力推荐
  • vue--为什么data属性必须是一个函数
  • yii2权限控制rbac之rule详细讲解
  • 分享几个不错的工具
  • 移动端 h5开发相关内容总结(三)
  • 树莓派用上kodexplorer也能玩成私有网盘
  • 微龛半导体获数千万Pre-A轮融资,投资方为国中创投 ...
  • ​queue --- 一个同步的队列类​
  • ​二进制运算符:(与运算)、|(或运算)、~(取反运算)、^(异或运算)、位移运算符​
  • ​如何防止网络攻击?
  • # 透过事物看本质的能力怎么培养?
  • (1/2) 为了理解 UWP 的启动流程,我从零开始创建了一个 UWP 程序
  • (2)(2.10) LTM telemetry
  • (C++17) optional的使用
  • (Java实习生)每日10道面试题打卡——JavaWeb篇
  • (八)c52学习之旅-中断实验
  • (第二周)效能测试
  • (附源码)计算机毕业设计SSM保险客户管理系统
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第3章 信息系统治理(一)
  • (四)【Jmeter】 JMeter的界面布局与组件概述
  • ../depcomp: line 571: exec: g++: not found
  • .NET C#版本和.NET版本以及VS版本的对应关系
  • .Net的DataSet直接与SQL2005交互
  • .net专家(高海东的专栏)
  • .Net组件程序设计之线程、并发管理(一)
  • /usr/lib/mysql/plugin权限_给数据库增加密码策略遇到的权限问题
  • [ vulhub漏洞复现篇 ] AppWeb认证绕过漏洞(CVE-2018-8715)
  • [acwing周赛复盘] 第 69 场周赛20220917
  • [Angular 基础] - 数据绑定(databinding)
  • [BZOJ1089][SCOI2003]严格n元树(递推+高精度)
  • [bzoj1324]Exca王者之剑_最小割
  • [C/C++]数据结构 深入挖掘环形链表问题
  • [C++] new和delete
  • [dfs] 图案计数