2019独角兽企业重金招聘Python工程师标准>>>

题目:统计一个数字在排序数组中出现的次数。例如输入排序数组为

{1,2,3,3,,3,3,4,5}和数字3,由于3在这个数组中出现了4次,因此输出4

既然输入的数组是排序的,那么我们很自然的想到利用二分查找算法。在 题目给出的例子中,我们可以先用二分查找算法找到第一个3.由于3可能出现多次,因此我们找到的3的左右两遍可能都是3,于是我们在找到3的左右两边顺序 扫描,分别找出第一个3和最后一个3.因为要查找的数字在长度为n的数组中可能很出现O(n)次,所以顺序扫描的时间复杂度为O(n)。因此这种算法的效 率和直接从头到尾顺序扫描整个数组统计3出现的次数的方法是一样的。显然,面试官是不会满意这种算法,它会提示我们还有更快的算法。

接下来我们思考如何更好的利用二分查找算法。假设我们统计数字k在排序数组中出现的次数。在前面的算法的时间主要消耗在如何确定重复出现的第一个k和最后一个k的位置上,有没有可以利用的二分查找算法直接找到第一个k和最后一个k。

我们先分析如何利用二分查找在数组中找到第一个k,二分查找算法总是 先拿数组的中间的数字和k做比较。如果中间的数字比k大,那么k只能出现在数组的前半段,下一轮我们旨在数组的前半段查找就可以了。如果中间的数字比k 小,那么k只能出现在数组的后半段,下一轮我们只在数组的后半段查找就可以了。如果中间的数字和k相等呢?我们先判断这个数字是不是第一个k。如果位于中 间数字的前面一个数字不是k,此时中间的数字刚好就是第一个k。如果中间的数字的前面一个数字也是k,也就是说第一个k肯定在数组的前半段,下一轮我们仍 然需要在数组的前半段查找。

同理我们利用上面的思路找到最后一个k。

找到第一个k和最后一个k后就可以知道k出现的次数了,

实现代码如下:

package cglib;

public class jiekou {

//递归找到排序数组中第一个k
     private int getFirstK(int[] arr,int k,int left,int right){  
            if(left > right)  
                return -1;  
            int middleIndex = (left+right)/2;  
            int middleData = arr[middleIndex];  
            if(middleData == k){ //如果中位数等于k
                if((middleIndex >0 && arr[middleIndex -1]!=k)|| middleIndex == 0)  
                    return middleIndex; //中位数是第一个k  
                else  
                    right = middleIndex -1;  //中位数不是第一个k,则往中位数左边找第一个k
            }  
            else if(middleData > k)//中位数大于k ,则第一个k肯定在中位数左边
                right = middleIndex -1;  //往中位数左边找第一个k
            else  
                left = middleIndex +1; //中位数小于k ,则第一个k肯定在中位数右边
            return getFirstK(arr,k,left,right); //递归
        }  
        //相同的思路,递归找到最后的一个k
        private int getLastK(int[] arr,int k,int left,int right){  
            if(left > right)  
                return -1;  
            int middleIndex = (left + right)/2;  
            int middleData = arr[middleIndex];  
            if(middleData == k){  
                if((middleIndex <arr.length -1 && arr[middleIndex+1]!=k) || middleIndex ==arr.length-1)  
                    return middleIndex;  
                else  
                    left = middleIndex+1;  //中位数不是最后一个k,则往中位数右边找最后个k
            }  
            else if(middleData <k){//中位数小于k,则往中位数右边找最后一个k  
                left = middleIndex +1;  
            }else  
                right = middleIndex -1;  //中位数大于k,则往中位数左边找最后一个k
            return getLastK(arr,k,left,right); //递归  
        }  
      //计算出k在数组中出现的次数
        public int getNumberOfK(int[] arr,int k){  
            int number = 0;  
            if(arr.length >0){  
                int first = getFirstK(arr,k,0,arr.length-1);  
                int last = getLastK(arr,k,0,arr.length -1);  
                if(first >-1 && last >-1)  
                    number =last-first+1;  
            }  
            return number;  
        }  
        public static void main(String[] args){  
            int[] arr= {1,2,3,3,3,3,4,5};  
            jiekou test = new jiekou();  
            System.out.println(test.getNumberOfK(arr, 3));  
        }
    }

输出:4

转载于:https://my.oschina.net/u/2822116/blog/726222

剑指Offer(Java版):数字在排序数组中出现的次数相关推荐

  1. 【牛客 - 剑指offer】JZ53 数字在升序数组中出现的次数 Java实现

    文章目录 剑指offer题解汇总 Java实现 本题链接 题目 方案一 暴力 方案二 二分(推荐) 剑指offer题解汇总 Java实现 https://blog.csdn.net/guligulig ...

  2. 剑指offer——面试题38:数字在排序数组中出现的次数

    剑指offer--面试题38:数字在排序数组中出现的次数 20180906整理 Solution1: [注意]先利用二分查找扎到一个值然后再顺序遍历的做法时间复杂度也是O(n)O(n)O(n),代码也 ...

  3. 【剑指offer】数字在排序数组中出现的次数

    转载请注明出处:http://blog.csdn.net/ns_code/article/details/27364557 题目描写叙述: 统计一个数字在排序数组中出现的次数. 输入: 每一个測试案例 ...

  4. 剑指offer java版 test3—从尾到头打印链表

    标题:剑指offer java版 test3-从尾到头打印链表 题目:输入一个链表,按链表从尾到头的顺序返回一个ArrayList. 解答:知识不够全面,用ArrayList做的 但是看到大佬们还可以 ...

  5. 《剑指offer》-- 调整数组顺序使奇数位于偶数前面、顺时针打印矩阵、数字在排序数组中出现的次数

    一.调整数组顺序使奇数位于偶数前面: 1.题目: 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于数组的后半部分,并保证奇数和奇数,偶数和偶数之 ...

  6. 剑指offer:数字在排序数组中出现的次数

    题目描述 统计一个数字在排序数组中出现的次数. 解题思路 暴力求解,没用到排序的已知条件. class Solution { public:int GetNumberOfK(vector<int ...

  7. 【剑指offer-Java版】38数字在排序数组中出现的次数

    数字在排序数组中出现的次数: 最简单粗暴的方法是O(n^2) 可能的解法可以采用二分,首先根据二分找到给定数字在数组中的位置,然后再左右二分,找到边界(第一个和最后一个),左右边界的差值就是出现次数 ...

  8. 《剑指offer》数字在排序数组中出现的次数

    题目:统计一个数字在排序数组中出现的次数. 解析:不要怀疑你的眼睛,也不用怀疑这题目难度,直接干! public class Solution {public int GetNumberOfK(int ...

  9. 剑指Offer值数字在排序数组中出现的次数

    题目描述 统计一个数字在排序数组中出现的次数. 解题思路 看到排序立马想到二分,利用STL的upper_bound与lower_bound处理即可.复杂度是O(log⁡N)O(\log N)O(log ...

最新文章

  1. Nature综述: 蓝藻水华的形成机理及防治动态
  2. doT.js 模板引擎的使用
  3. c盘python27文件夹可以删除嘛_电脑C盘那些文件夹可以删除
  4. jQuery Event.stopPropagation() 函数详解
  5. service能去调另外一个service吗_kubernetes的service和pod是如何关联的?
  6. jmeter 循环取值赋值给form_JMeter系列(三)逻辑控制器详解
  7. VSTO 得到Office文档的选中内容(Word、Excel、PPT、Outlook)
  8. 廖雪峰git学习(3)分支管理【待续】
  9. 为什么小角度的弧度值可以近似等于正切值或者正弦值?
  10. 51单片机两只老虎 c语言,基于51单片机的简易电子琴(两只老虎)
  11. python向excel隔行写数据_【Python】将每日新增数据写入Excel
  12. 【Android】安卓开发实战之软键盘设置
  13. 记住你跑得快,别人跑得更快。
  14. 《魂斗罗:归来》子弹中没中,没你想得那么简单!
  15. 小米手机只能显示e或者无服务器,手机出现E网怎么回事
  16. 甬矽电子科创板上市:年营收21亿募资11亿 市值122亿
  17. win11锁屏壁纸不更新,且全黑的解决办法
  18. MySQL中GROUP_CONCAT函数的使用,separator,将多行查询结果用特定字符串连接起来,适用于一对多
  19. 计算机日常故障DIY维修有哪些,计算机维护 DIY 完全手册
  20. C 语言调用 JNI_CreateJavaVM

热门文章

  1. 对话农民丰收节交易会-林裕豪:从玉农业2021新年贺词
  2. Spring JPA使用CriteriaBuilder动态构造查询
  3. mzy,struts学习(一)
  4. 学习ASP.NET Core Razor 编程系列五——Asp.Net Core Razor新建模板页面
  5. 使用Lombok简化你的代码
  6. js 导出到excel
  7. Maven构建Struts2框架的注意事项
  8. .NET特性(Attribute)应用一例
  9. 各国家分析之 古埃及非洲经济
  10. 不同国别的买家,谈判方式竟有如此大差异!