美文网首页互联网科技我用 LinuxLinux
小朋友学经典算法(17):分治法求众数

小朋友学经典算法(17):分治法求众数

作者: 海天一树X | 来源:发表于2019-03-19 11:23 被阅读9次

一组数据中,出现次数最多的数就叫这组数据的众数。
如果有两个或两个以上个数出现次数都是最多的,那么这几个数都是这组数据的众数。
如果所有数据出现的次数都一样,那么这组数据没有众数。

例1:1,2,3,3,4的众数是3。
例2:1,2,2,3,3,4的众数是2和3。
例3:1,2,3,4,5没有众数。

解法一:

#include <iostream>
#include <algorithm>
using namespace std;

int cnt[100000000];
int main()
{
    int n;
    cin >> n;
    int a[n];
    int maxCnt = 0;
    for(int i = 0; i < n; i++)
    {
        cin >> a[i];
        cnt[a[i]]++;
        if(maxCnt < cnt[a[i]])
        {
            maxCnt = cnt[a[i]];
        }
    }

    sort(a, a + n);

    bool same = true;
    for(int i = 0; i < n - 1; i++)
    {
        if(cnt[a[i]] != cnt[a[i + 1]])
        {
            same = false;
            break;
        }
    }

    if(same)
    {
        cout << "没有众数" << endl;
        return 0;
    }

    int i = 0;

    while(i < n)
    {
        if(cnt[a[i]] == maxCnt)
        {
            cout << "众数:" << a[i] << ",出现的次数:" << cnt[a[i]] << endl;
        }
        i += cnt[a[i]];
    }

    return 0;
}

解法二:分治法

以a = {1, 2, 2, 2, 3, 3, 5, 6, 6, 6, 6},其中间元素为a[5] = 3。往左边找等于3的最大位置,得到l = 4, a[l] = a[4] = 3;往右边找不等于3(即大于3)的最小位置,得到r = 6, a[r] = a[6] = 5。则可以知道,最中间的3出现的次数cnt = r - l = 2次,即maxCnt = 2次。
接着进行分治:考虑元素3左边的a[0] ~ a[l]部分和右边的a[6] ~ a[10]部分。对这两部分各自求cnt。得到左边部分有3个2,右边部分有4个6。则最终众数是6,出现了maxCnt = 4次。
在递归的过程中,如果左边或右边的元素数量小于maxCnt,那就不需要再递归了。这是递归的终止条件。

// 用分治法求众数
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<map>

using namespace std;

map<int, int> m;

// 左右两边与中间数相同的数的起始、终止界限
void split(int s[], int n, int &l, int &r)
{
    int mid = n/2;
    // 连续两个for求与s[mid]相同的数有多少个
    for(l=0; l<n; ++l)
    {
        if (s[l] == s[mid])
            break;
    }
    for(r=l+1; r<n; ++r)
    {
        if (s[r] != s[mid])
            break;
    }

}

// num 众数。 maxCnt 重数
void getMaxCnt(int &mid, int &maxCnt, int s[], int n)
{
    int l, r;
    split(s, n, l, r);  // 进行分割。这个函数是本程序的关键
    int num = n/2;
    int cnt = r-l;

    // 更新出现次数最多的数
    if (cnt > maxCnt)
    {
        maxCnt = cnt;
        mid = s[num];
        m.clear();
        m[mid] = maxCnt;
    }
    else if(cnt == maxCnt)
    {
        mid = s[num];
        m[mid] = maxCnt;
    }

    // l表示mid左边的个数。左边的个数必须大于 maxCnt才有必要搜寻
    if (l >= maxCnt)
    {
        getMaxCnt(mid, maxCnt, s, l);
    }

    // n-r表示mid右边的个数, 右边数组的起始地址要变更
    if (n-r >= maxCnt)
    {
        getMaxCnt(mid, maxCnt, s+r, n-r);
    }
}

int main(void)
{
    int s[] = {1, 2, 2, 2, 3, 3, 5, 6, 6, 6, 6};
    //int s[] = {20, 20, 30, 30};
    int n = sizeof(s)/sizeof(s[0]);
    sort(s, s + n);

    int maxCnt = 0;
    int num = 0;
    getMaxCnt(num, maxCnt, s, n);

    map<int, int>::iterator it;
    int sum = 0;
    for(it = m.begin(); it != m.end(); it++)
    {
        sum += it->second;
    }
    if(sum == n) // 唯一没有众数的情景
    {
        cout << "没有众数" << endl;
    }
    else
    {
        for(it = m.begin(); it != m.end(); it++)
        {
            cout << "众数:" << it->first << ",出现次数:" << it->second << endl;
        }
    }


    return 0;
}

少儿编程、算法咨询请加微信307591841或QQ群581357582


信息学竞赛公众号.jpg

相关文章

  • 小朋友学经典算法(17):分治法求众数

    一组数据中,出现次数最多的数就叫这组数据的众数。如果有两个或两个以上个数出现次数都是最多的,那么这几个数都是这组数...

  • 归并排序

    阅读经典——《算法导论》02 不同算法中往往蕴含着通用的思想,分治法就是最常用的一种。 分治法使用递归的方式,将原...

  • LeetCode 169 求众数 Majority Elemen

    有关递归与分治的做题笔记,Python实现 169. 求众数 Majority Element LeetCodeC...

  • 12 基本排序算法:归并排序

    归并排序 原理 归并排序思想 该算法采用经典的分治(divide-and-conquer)策略(分治法将问题分(d...

  • Divide and Conquer

    算法之 分治法 Divide and Conquer 分治法: 分治法的设计思想是:将一个难以直接解决的大问题,分...

  • 分治法(Divide-and-Conquer Algorithm

    上次给大家带来了分治法的基本介绍和基本思想,今天我们继续来看分治算法的几个经典例子。 **01 ** 快速排序 1...

  • 分治法,动态规划及贪心算法区别

    原文:分治法,动态规划及贪心算法区别 1.分治法 分治法(divide-and-conquer):将原问题划分成n...

  • [算法导论]归并排序

    时间复杂度 《算法导论》2.3.1 分治法。 归并排序采用了分治法的递归排序。分治法:分解子问题,解决子问题,合并...

  • 归并排序

    1、分治法 归并排序是完全遵循分治策略的排序算法。什么是分治法? 分治法,即将原问题分解为几个规模较小的子问题,递...

  • 算法导论第2.3章 - 分治算法

    分治算法 递归:算法一次或多次递归地调用其自身已解决紧密相关的若干子问题。这些算法遵循分治法的思想。 分治算法三个...

网友评论

    本文标题:小朋友学经典算法(17):分治法求众数

    本文链接:https://www.haomeiwen.com/subject/becgmqtx.html