题目描述:

数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为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)。

快排的代码如下:

void Qsort(int begin,int end){
int middle;
if(begin < end){
middle = Patition(begin,end); Qsort(begin,middle -);
Qsort(middle+,end);
}
}
int Patition(int begin,int end){
int middle = gArr[begin];
while(begin < end){
while(begin < end && gArr[end] >= middle)
end--;
swap(begin,end); while(begin < end && gArr[begin] <= middle)
begin++;
swap(begin,end);
}
return begin;
}
void swap(int begin,int end){
int tmp = gArr[end];
gArr[end] = gArr[begin];
gArr[begin] = tmp;
}

全部代码:

#include <stdio.h>
#include <stdlib.h>
#define MAXSIZE 100001
int gArr[MAXSIZE] = {};
void Qsort(int begin,int end);
void swap(int begin,int end);
int Patition(int begin,int end);
int main(){
int n,i,middle,count;
while(scanf("%d",&n)!=EOF && n> && n <= ){
for(i=;i<n;i++){
scanf("%d",&gArr[i]);
}
Qsort(,n-);
middle = gArr[n/];
count = ;
for(i=;i<n;i++){
if(middle == gArr[i])
count++;
}
if(count > n/)
printf("%d\n",middle);
else
printf("-1\n");
}
}
void Qsort(int begin,int end){
int middle;
if(begin < end){
middle = Patition(begin,end); Qsort(begin,middle -);
Qsort(middle+,end);
}
}
int Patition(int begin,int end){
int middle = gArr[begin];
while(begin < end){
while(begin < end && gArr[end] >= middle)
end--;
swap(begin,end); while(begin < end && gArr[begin] <= middle)
begin++;
swap(begin,end);
}
return begin;
}
void swap(int begin,int end){
int tmp = gArr[end];
gArr[end] = gArr[begin];
gArr[begin] = tmp;
} /**************************************************************
Problem: 1370
User: xhalo
Language: C
Result: Accepted
Time:800 ms
Memory:1304 kb
****************************************************************/

另外一种思路

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

#include <stdio.h>
#include <stdlib.h>
#define MAXSIZE 100000
typedef struct flag{
int data;
int counter;
}Flag;
typedef struct fArr{
struct flag arr[MAXSIZE];
}FArr;
int gArr[MAXSIZE] = {};
int gnum;
int main(){
int n,i,max,maxNum;
while(scanf("%d",&n)!=EOF && n> && n <= ){
FArr *a = (FArr *)malloc(sizeof(FArr));
gnum = ;
max = -;
maxNum = -;
for(i=;i<n;i++){
scanf("%d",&gArr[i]);
}
for(i=;i<n;i++){
int j = ;
while(j<gnum){
if(a->arr[j].data == gArr[i])
break;
j++;
}
if(gnum != && j != gnum){
a->arr[j].counter++;
}else{
a->arr[gnum].data = gArr[i];
a->arr[gnum++].counter = ;
}
}
for(i=;i<gnum;i++){
if(max < a->arr[i].counter){
max = a->arr[i].counter;
maxNum = a->arr[i].data;
}
}
if(max > n/)
printf("%d\n",maxNum);
else
printf("-1\n");
}
}
/**************************************************************
Problem: 1370
User: xhalo
Language: C
Result: Time Limit Exceed
****************************************************************/

剑指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. HDU1171——Big Event in HDU(母函数)

    Big Event in HDU DescriptionNowadays, we all know that Computer College is the biggest department in ...

  2. Android view的requestLayout()

    public void requestLayout () Since: API Level 1 Call this when something has changed which has inval ...

  3. C#如何在派生类中不显示父类的一些属性以及TypeDescriptor使用

    public SonClass:FatherClass { 定义属性 .... } Type thisType = typeof(SonClass);方法一: PropertyInfo[] pis = ...

  4. C#中this在构造函数时的使用

    今天编程的时候,想要用this来处理构造函数,想了半天没有想起来 后来找了自己以前记录的 http://www.cnblogs.com/chucklu/p/4842766.html public Cu ...

  5. Mongodb的范式化和反范式化

    如果是涉及到一对多的数据格式,可使用文档引用范式化数据. 在一个,User对象中,如果涉及到工作信息或者联系地址的,这些信息会频繁的进行访问,可使用嵌入式文档对数据进行反范式化.

  6. CodeForces Round #286 Div.2

    A. Mr. Kitayuta's Gift (枚举) 题意: 给一个长度不超过10的串,问能否通过插入一个字符使得新串成为回文串. 分析: 因为所给的串很多,所以可以枚举 “在哪插入” 和 “插入什 ...

  7. UVa 1213 (01背包变形) Sum of Different Primes

    题意: 选择K个质数使它们的和为N,求总的方案数. 分析: 虽然知道推出来了转移方程, 但还是没把代码敲出来,可能基本功还是不够吧. d(i, j)表示i个素数的和为j的方案数,则 d(i, j) = ...

  8. Zend Framework XML外部实体和安全绕过漏洞

    漏洞版本: Zend Framework 1.x 漏洞描述: Bugtraq ID:66358 Zend Framework是一款开放源代码的PHP5开发框架实现. Zend Framework存在多 ...

  9. oracle 字段上下两条记录的相减

    SELECT T.ID  ,BALANCE,nvl(lag (BALANCE,1) over (order by T.ID ) ,0) FROM  AN T ORDER BY T.ID [转]orac ...

  10. 装饰器模式(Decorator)

    转自http://blog.csdn.net/hust_is_lcd/article/details/7884320 1.认识装饰器模式 装饰模式能够实现动态的为对象添加功能,是从一个对象外部来给对象 ...