「LG2397」yyy loves Maths VI (mode)

题目描述


他让redbag找众数
他还特意表示,这个众数出现次数超过了一半
一共n个数,而且保证有
n<=2000000
而且每个数<2^31-1

输入输出格式

输入格式:

第一行一个整数n
第二行n个整数

输出格式:

一行,这个众数

输入输出样例

5
2 3 3 3 3

3

题解

Solution1

第一个思路还是比较难♂好想,对于每一个数取二进制的每一位计数,然后如果有一位上的个数比n/2大,就加进答案。最后和就是ans。

#include<stdio.h>
#include<stdlib.h>
inline int gi(){
    int sum=0,f=1;char ch=getchar();
    while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0' && ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
    return sum*f;
}
int have[40];
int main(){
    int n,m;
    scanf("%d",&n);
    for(register int i=1;i<=n;i++){
        scanf("%d",&m);
        for(register int j=0;(1<<j)<=m;j++){
            if((1<<j)&m)have[j]++;
        }
    }
    int ans=0;
    for(register int i=0;i<=30;i++){
        if(have[i]>n/2)ans+=1<<i;
    }
    printf("%d\n",ans);
    return 0;
}

Solution2

第二个思路用到的是摩尔投票法,这种方法一般适用于求众数时众数数量比一半多.
首先我们可以证明这样子的数必然只有一个,如果有两个总共就不只n个数了.然后再仔细想一想,就发现这道题目比较愚蠢良心,用一个变量记录当前里面的数,如果相同就累计,不同就减掉.这样重复下去,剩下的数必然是众数.

#include<stdio.h>
#include<stdlib.h>
inline int gi(){
    int sum=0,f=1;char ch=getchar();
    while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0' && ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
    return sum*f;
}
int main(){
    int n,m;
    scanf("%d",&n);int now=0,in;
    for(register int i=1;i<=n;i++){
        scanf("%d",&m);
        if(!now || m==in){now++;in=m;}
        else now--;
    }
    printf("%d\n",in);
    return 0;
}

posted @ 2018-07-21 23:32  cj_gjh  阅读(130)  评论(0编辑  收藏  举报