题目描述:

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

输入:

每个测试案例包括2行:

第一行输入一个整数n(1<=n<=100000),表示数组中元素的个数。

第二行输入n个整数,表示数组中的每个元素,这n个整数的范围是[1,1000000000]。

输出:

对应每个测试案例,输出出现的次数超过数组长度的一半的数,如果没有输出-1。

样例输入:
样例输出:

解题思路:

  有两种思路。先说在这道题目上能成功的:

  我们首先对数组进行排序,因为要找出数目超过一半的数,因此,如果存在,那么这个数组的中间值肯定是这个数。比如

  1 2 2 2 1 或者 1 1 2 2 2 或者 2 2 2 3 3,中间的肯定是我们要找的数。

  而如果不存在,那么进行一次O(n)的扫描即可。因此我们的算法时间复杂度为快排+一次遍历,O(nlogn)+O(n)。

快排的代码如下:

  1. void Qsort(int begin,int end){
  2. int middle;
  3. if(begin < end){
  4. middle = Patition(begin,end);
  5.  
  6. Qsort(begin,middle -);
  7. Qsort(middle+,end);
  8. }
  9. }
  10. int Patition(int begin,int end){
  11. int middle = gArr[begin];
  12. while(begin < end){
  13. while(begin < end && gArr[end] >= middle)
  14. end--;
  15. swap(begin,end);
  16.  
  17. while(begin < end && gArr[begin] <= middle)
  18. begin++;
  19. swap(begin,end);
  20. }
  21. return begin;
  22. }
  23. void swap(int begin,int end){
  24. int tmp = gArr[end];
  25. gArr[end] = gArr[begin];
  26. gArr[begin] = tmp;
  27. }

全部代码:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #define MAXSIZE 100001
  4. int gArr[MAXSIZE] = {};
  5. void Qsort(int begin,int end);
  6. void swap(int begin,int end);
  7. int Patition(int begin,int end);
  8. int main(){
  9. int n,i,middle,count;
  10. while(scanf("%d",&n)!=EOF && n> && n <= ){
  11. for(i=;i<n;i++){
  12. scanf("%d",&gArr[i]);
  13. }
  14. Qsort(,n-);
  15. middle = gArr[n/];
  16. count = ;
  17. for(i=;i<n;i++){
  18. if(middle == gArr[i])
  19. count++;
  20. }
  21. if(count > n/)
  22. printf("%d\n",middle);
  23. else
  24. printf("-1\n");
  25. }
  26. }
  27. void Qsort(int begin,int end){
  28. int middle;
  29. if(begin < end){
  30. middle = Patition(begin,end);
  31.  
  32. Qsort(begin,middle -);
  33. Qsort(middle+,end);
  34. }
  35. }
  36. int Patition(int begin,int end){
  37. int middle = gArr[begin];
  38. while(begin < end){
  39. while(begin < end && gArr[end] >= middle)
  40. end--;
  41. swap(begin,end);
  42.  
  43. while(begin < end && gArr[begin] <= middle)
  44. begin++;
  45. swap(begin,end);
  46. }
  47. return begin;
  48. }
  49. void swap(int begin,int end){
  50. int tmp = gArr[end];
  51. gArr[end] = gArr[begin];
  52. gArr[begin] = tmp;
  53. }
  54.  
  55. /**************************************************************
  56. Problem: 1370
  57. User: xhalo
  58. Language: C
  59. Result: Accepted
  60. Time:800 ms
  61. Memory:1304 kb
  62. ****************************************************************/

另外一种思路

  我们对每个元素进行统计。但是在记录统计时,有个麻烦处,就是如何从记录数组中查找到我们要记录的元素。下面的代码在数据量很大,我猜测是100000个不重复的点,因此进行遍历时超时了。不过在小数据量时,还是可以的:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #define MAXSIZE 100000
  4. typedef struct flag{
  5. int data;
  6. int counter;
  7. }Flag;
  8. typedef struct fArr{
  9. struct flag arr[MAXSIZE];
  10. }FArr;
  11. int gArr[MAXSIZE] = {};
  12. int gnum;
  13. int main(){
  14. int n,i,max,maxNum;
  15. while(scanf("%d",&n)!=EOF && n> && n <= ){
  16. FArr *a = (FArr *)malloc(sizeof(FArr));
  17. gnum = ;
  18. max = -;
  19. maxNum = -;
  20. for(i=;i<n;i++){
  21. scanf("%d",&gArr[i]);
  22. }
  23. for(i=;i<n;i++){
  24. int j = ;
  25. while(j<gnum){
  26. if(a->arr[j].data == gArr[i])
  27. break;
  28. j++;
  29. }
  30. if(gnum != && j != gnum){
  31. a->arr[j].counter++;
  32. }else{
  33. a->arr[gnum].data = gArr[i];
  34. a->arr[gnum++].counter = ;
  35. }
  36. }
  37. for(i=;i<gnum;i++){
  38. if(max < a->arr[i].counter){
  39. max = a->arr[i].counter;
  40. maxNum = a->arr[i].data;
  41. }
  42. }
  43. if(max > n/)
  44. printf("%d\n",maxNum);
  45. else
  46. printf("-1\n");
  47. }
  48. }
  49. /**************************************************************
  50. Problem: 1370
  51. User: xhalo
  52. Language: C
  53. Result: Time Limit Exceed
  54. ****************************************************************/

剑指OFFER之数组中出现次数超过一半的数字(九度OJ1370)的更多相关文章

  1. 剑指Offer:数组中出现次数超过一半的数字【39】

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

  2. 剑指 Offer 39. 数组中出现次数超过一半的数字 + 摩尔投票法

    剑指 Offer 39. 数组中出现次数超过一半的数字 Offer_39 题目描述 方法一:使用map存储数字出现的次数 public class Offer_39 { public int majo ...

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

    剑指 Offer 39. 数组中出现次数超过一半的数字 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 你可以假设数组是非空的,并且给定的数组总是存在多数元素. 示例 1: 输入: [ ...

  4. 力扣 - 剑指 Offer 39. 数组中出现次数超过一半的数字

    题目 剑指 Offer 39. 数组中出现次数超过一半的数字 思路1(排序) 因为题目说一定会存在超过数组长度一半的一个数字,所以我们将数组排序后,位于length/2位置的一定是众数 代码 clas ...

  5. 【剑指Offer】数组中出现次数超过一半的数字 解题报告(Python)

    [剑指Offer]数组中出现次数超过一半的数字 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-inter ...

  6. 【Java】 剑指offer(39) 数组中出现次数超过一半的数字

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如 ...

  7. Go语言实现:【剑指offer】数组中出现次数超过一半的数字

    该题目来源于牛客网<剑指offer>专题. 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组 ...

  8. 剑指Offer 28. 数组中出现次数超过一半的数字 (数组)

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

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

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

随机推荐

  1. 【ZOJ】2112 Dynamic Rankings

    树状数组套主席树模板题目. /* 2112 */ #include <iostream> #include <sstream> #include <string> ...

  2. [转] android自动化测试之MonkeyRunner使用实例(三)

    一.使用CMD命令打开模拟器 运行monkeyrunner之前必须先运行相应的模拟器或连上设备,不然monkeyrunner无法连接设备. 1.1  用Elipse打开Android模拟器或在CMD中 ...

  3. 使用Spring框架的12个开源项目

    使用Spring框架的12个开源项目 http://www.csdn.net/article/2013-10-14/2817176-open-source-projects-that-use-spri ...

  4. poj2186Popular Cows(强连通分量)

    http://poj.org/problem?id=2186 用tarjan算出强连通分量的个数 将其缩点 连成一棵树  则题目所求即变成求出度为0 的那个节点 在树中是唯一的 即树根 #includ ...

  5. Linux 模拟 鼠标 键盘 事件

    /************************************************************************ * Linux 模拟 鼠标 键盘 事件 * 说明: ...

  6. 教你用Java安全有效的实现两星期内自动登陆功能-Session

    现在很多网站都有为用户保存登陆信息(即保存Cookie)的功能,当用户下一次进入网站时,可以帮助用户自动登陆,使网站显得更加友好.笔者通过研究ACEGI项目的自动登陆源码,编写了一个安全有效的实现两星 ...

  7. Can't find file: './mysql/plugin.frm' (errno: 13)[mysql数据目录迁移错位]错误解决

    大概需要4个步骤,其中第1步通过service mysql stop停止数据库,第4步通过service mysql start启动数据库. 第2步移动数据文件,不知道是否为Ubuntu智能的原因,移 ...

  8. ImagePacker

    [下载]:ImagePacker 0.0.2 [作用]:将多个图片打包到指定大小的图片中,并输出配置文件.类似于 TexturePacker,不过当前只支持 Starling 格式输出(因为我现在使用 ...

  9. VTK三维重建(1)-使用VTK读取DICOM,并动态输出

    [效果显示] 将脚部骨骼扫描的CT照片进行的连续读取, 运行结果存为了两个动态gif, 不知道能不能正常显示 [程序实现] int main(int argc, char* argv[]) { // ...

  10. [NOI导刊2011]影像之结构化特征

    问题描述 在影像比对中,有一种方法是利用影像中的边缘(edge)资讯,计算每个边缘资讯中具有代表性的结构化特征,以作为比对两张影像是否相似的判断标准.Water-filling方法是从每个边缘图的一个 ...