看题目:

数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。

我的直接思路:

用map计数,简单直接,遍历一次数组,用hashmap记录,key为int值,value为出现次数;

第二次再用map.entrySet找出有没value大于数组长度一般的entry,有的话返回它的key。

时间复杂度也是2n而已,这个方法时间复杂度是O(n)空间复杂度也是O(n)

代码实现:

  1. /*方法1
  2. 蛮力,遍历一次,用一个map来记录
  3. 第二次遍历把出现次数大于length/2的那个值找出来
  4. */
  5.  
  6. public int MoreThanHalfNum_Solution(int [] array) {
  7. int targetCount = array.length / 2;
  8.  
  9. //key为数组中的值,value为出现的次数
  10. HashMap<Integer,Integer> map = new HashMap<Integer,Integer>();
  11. for(int temp : array) {
  12. if(map.get(temp) == null)map.put(temp, 1);
  13. else map.put(temp, map.get(temp) + 1);
  14. }
  15. Set<Entry<Integer,Integer>> entrySet = map.entrySet();
  16. for(Entry<Integer,Integer> entry : entrySet) {
  17. if(entry.getValue() > targetCount)return entry.getKey();
  18. }
  19. return 0;
  20. }

方法2:中位数

如果数组是排好序的,就好解决了:
  如果排好序,然后又存在这样的数字的话,那么它的值肯定和数组中间那个值是一样的!!(比如:1,2,2,2,3;或2,2,2,3,4;或2,3,4,4,4等等)
 
所以我们只需要排序后,一次遍历,访问每个元素都和中间值比较,只要相等计数器就加一,遍历完后如果计数器大于数组长度一半就返回那个中间元素就好。
 
排序最快是O(Nlogn),然后加上遍历是O(n),空间复杂度是O(1)
 
代码里直接用的是Arrays的sort方法,它的实现没记错是O(Nlogn)的快排。
看代码:
  1. /*方法2
  2. 排序后中位数法
  3. 如果有个数字出现的次数大于数组长度的一半,那么这个数组排序后,它的中位数一定是这个数字
  4. */
  5.  
  6. public int MoreThanHalfNum_Solution(int [] array) {
  7. Arrays.sort(array);
  8. int count = 0, middleNum = array[array.length / 2];
  9. for(int temp : array) {
  10. if(temp == middleNum)count++;
  11. }
  12. if(count > array.length / 2)return middleNum;
  13. else return 0;
  14. }

方法3:——快排思路

方法2中,我们排序是为了找出中位数,那么如果可以更快地找出中位数就不用排序了。

借鉴快速排序算法,其中的Partition()方法是一个最重要的方法,该方法返回一个index,能够保证index位置的数是已排序完成的,在index左边的数都比index所在的数小,在index右边的数都比index所在的数大。那么本题就可以利用这样的思路来解。

通过Partition()返回index,如果index==mid,那么就表明找到了数组的中位数如果index<mid,表明中位数在[index+1,end]之间;如果index>mid,表明中位数在[start,index-1]之间。直到最后求得index==mid循环结束。

 
说白了就是找中位数这一步,我们不选择直接排序数组,而是用快排的Partition方法
 
该算法的时间复杂度为O(n),空间复杂度为O(1)
 
看代码:
  1. /*方法3
  2. 利用快排的思想
  3. */
  4.  
  5. public int MoreThanHalfNum_Solution(int [] array) {
  6. if(array.length <= 0)return 0;
  7.  
  8. int begin = 0, end = array.length - 1, middle = array.length / 2;
  9. int partition = partition(array, begin, end);
  10.  
  11. while(partition != middle) {
  12. if(partition > middle) {//中位数在partition的左边
  13. partition = partition(array, begin, partition - 1);
  14. } else {//中位数在partition右边
  15. partition = partition(array, partition + 1, end);
  16. }
  17. }
  18.  
  19. //找出中位数了,看这个中位数出现的次数是否符合要求
  20. int count = 0, middleKey = array[middle];
  21. for(int temp : array) {
  22. if(temp == middleKey)count++;
  23. }
  24.  
  25. if(count > array.length / 2)return array[middle];
  26. else return 0;
  27. }
  28.  
  29. //这个方法是以第一个元素为基准,然后进行划分,划分后比基准元素小的数字都在它左边,比它大的数字都在它右边
  30. 返回划分后,这个元素的新index//
  31. private int partition(int[] a, int begin, int end) {
  32. int key = a[begin];
  33.  
  34. int i = begin, j = end;
  35. while(i < j) {
  36. while(i < j && a[j] >= key)j--;
  37. while(i < j && a[i] <= key)i++;
  38. swap(a, i, j);
  39. }
  40. swap(a, begin, i);
  41. return i;
  42. }
  43.  
  44. //交换数字函数,传入数组还有要交换的两个数字的index//
  45. private void swap(int[] a, int first, int second) {
  46. int temp = a[first];
  47. a[first] = a[second];
  48. a[second] = temp;
  49. }
方法4——阵地攻守思想
第一个数字作为第一个士兵,守阵地;count = 1;
遇到相同元素,count++;
遇到不相同元素,即为敌人,同归于尽,count--;当遇到count为0的情况,又以新的i值作为守阵地的士兵,继续下去,同时count更新为1.
到最后还留在阵地上的士兵,有可能是主元素。
再加一次循环,记录这个士兵的个数看是否大于数组一般即可。
 
这个方法也是主要因为考虑到:题目中要找的数字出现的次数超过数组长度的一半,也就是说它出现的次数比其他所有数字出现的次数的和还要多。,那么肯定最后留下来的是那个出现超过一半的。
 
该方法时间复杂度O(N),空间复杂度O(1)
 
 
实现的时候就维护两个变量:一个是数组中的一个数字,一个是次数。

当我们遍历到下一个数字的时候,

如果下一个数字和当前我们保存的数字相同,则次数加 1;

如果和当前我们保存的数字不同,则次数减 1;

当次数减到 0 的时候,我们将保存的数字改为当前遍历所处的位置,并将次数更改为 1。

看代码:

  1. /*方法4
  2. 阵地攻守思想,其实和牛客上面那个什么“用户分形叶”的思路一样的,不同实现而已
  3. */
  4. public int MoreThanHalfNum_Solution(int [] array) {
  5. //int count = 1;
  6. int count = 0;//这里我们设初始的count为0好了,因为利用java的foreach语法是要从第一个key开始遍历的,那么就是碰到第一个开始,设为1
  7. int key = array[0];
  8. for(int temp : array) {
  9. if(temp == key)count++;
  10. else if(count > 0)count--;
  11. else {//count==0的情况,这个时候把key换成现在这个元素,并把count设为1,意思是这是第一次碰到这个元素
  12. key = temp;
  13. count = 1;
  14. }
  15. }
  16.  
  17. //判断这个得到的key是不是符合要求
  18. int count2 = 0;
  19. for(int temp : array) {
  20. if(temp == key)count2++;
  21. }
  22. if(count2 > array.length / 2)return key;
  23. else return 0;
  24. }

剑指Offer——数组中出现次数超过一半的数字——一题多解的更多相关文章

  1. python剑指offer数组中出现次数超过一半的数字

    题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...

  2. 剑指offer——数组中出现次数超过一半的数字(c++)

    题目描述数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2.如 ...

  3. 剑指Offer——数组中出现次数超过一半的数字

    题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2 ...

  4. 用js刷剑指offer(数组中出现次数超过一半的数字)

    题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...

  5. 剑指Offer-28.数组中出现次数超过一半的数字(C++/Java)

    题目: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2.如 ...

  6. 剑指offer--11.数组中出现次数超过一半的数字

    unique(), count()函数好用 ---------------------------------------------------------------------- 时间限制:1秒 ...

  7. 1-剑指offer: 数组中出现次数超过一半的数字

    题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...

  8. 剑指offer-数组中出现次数超过一半的数字

    题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...

  9. Leetcode - 剑指offer 面试题29:数组中出现次数超过一半的数字及其变形(腾讯2015秋招 编程题4)

    剑指offer 面试题29:数组中出现次数超过一半的数字 提交网址: http://www.nowcoder.com/practice/e8a1b01a2df14cb2b228b30ee6a92163 ...

随机推荐

  1. QE名词解释以及相关文章链接

    百科: http://baike.baidu.com/link?url=ho-aUG2rZwgjx75rwFu5b3XoQnsuJMj9GrJEuaZxnakg19ofO13mrXCMi9_JZ_VY ...

  2. office2016_windows永久激活查看方法

    YC7N8-G7WR6-9WR4H-6Y2W4-KBT6X 首先要保证你安装了 百云址:http://pan.baidu.com/share/home?uk=4011207371 如果你是win8,w ...

  3. Python:map()、reduce()、filter()的区别

    文章转于:https://blog.csdn.net/goupper1991/article/details/49803355 原文博主:https://blog.csdn.net/goupper19 ...

  4. springMVC绑定json参数之一

    一.SpringMVC @RequestBody接收Json对象字符串 以前,一直以为在SpringMVC环境中,@RequestBody接收的是一个Json对象,一直在调试代码都没有成功,后来发现, ...

  5. asp中实现lable自动换行

    asp中实现lable自动换行 因为在用Label标签显示内容时,内容太多,想实现自动换行.我们知道在WINFORM中程序中,有一个属性是AUTOSIZE 改成FALSE 是可以实现的.但是在ASP. ...

  6. kvm ip查看

    1.virsh --list(查看有哪些服务器) 2.virsh dumpxml 虚拟机名称 查看服务器对应的mac地址 3.然后再宿主机上arp -a 查看对应的mac地址对应的ip

  7. 在Action获取Scope对象

    引言:在前面的Action操作中,关键就是Action中的exectue方法,但是此方法并没有request.session.application等对象作为参数,自然就不能利用这些对象来操作.下面我 ...

  8. glib-2.40编译安装

    1 安装glib库所需要的依赖库: libffi-.tar.gz glib-.tar.xz 安装依赖库libffi: tar xf libffi-.tar.gz cd libffi- ./config ...

  9. java类什么时候初始化?

    Java虚拟机规范中并没有进行强制玉树什么情况下需要开始类加载过程.但是对于初始化阶段,虚拟机规范则是严格规定了有且仅有5种情况必须立即对类进行“初始化”(而加载,验证,准备自然需要在此之前开始): ...

  10. es6基础系列四--字符串的拓展

    1 for...of 字符串的遍历接口 for(let i of "abc"){ console.log(i); } // a // b // c 2 includes 是否包含某 ...