计数排序

计数排序

原理

    计数排序的基本思想是:对于每一个输入元素x,确定小于x的元素个数。根据这些信息,就能准确的将每个数字放在正确的位置上。

    在排序的过程中,除了输入数组A[n]之外,还需要一个记录结果的数组B[n]和一个记录小于x数的个数数组C[num_max]。A与B的数组等长,而C数组的长度则为n个数中的最大者。

这里写图片描述

    上图为计数排序的一系列过程,下面依次做分析:

  1. (a)中显示了原始数组A,数组C[n]中元素c[x]即与下标x相等的元素个数。
  2. (b)中对计数数组C做了进一步的处理,即C[x+1] = c[x+1] + c[x],将前一项和本项之和赋值给本项,用来快速的统计小于等于x的数的个数。
  3. (c)中开始依据C数组将数字的序列填写如结果数组B。进行两步操作,即B[C[A[j]]] = A[j]和C[A[j]] = C[A[j]] - 1。第一步操作的意思即依照C数组中的信息,将A数组中的每个元素挨个填入,第二步操作是为了防止有相等的元素产生,需要向前挪一位。比如图(c)中填入的第一个为元素3,在C数组中显示小于等于3的元素有7个,那我们就将3填入B数组中的7号位,为了防止下一个3填入相同的位置,将3的计数减少1,向前挪一位,下一个3填入的时候就到了6号位。
  4. (d)重复(c)图中的两步操作,正在填写元素0.
  5. (e)重复(c)图中的操作,正在填写第二个3。
  6. 图(f)为已经填写完毕的结果数组B。

C语言实现

int max_num(int *a, int count)
{
    int max = a[0];
    for (int i = 1; i < count; i++)
    {
        if (a[i] > max)
            max = a[i];
    }
    return max+1;
}

int* counting_sort(int* a, int count)
{
    int max, *b, *c;
    max = max_num(a, count);
    b = (int*)malloc(count * 2);
    c = (int*)malloc(max * 2);

    //统计A中与C下标相等的元素,即图(a)
    for (int i = 0; i < max; i++)
    {
        int c_count = 0;
        for (int j = 0; j < count; j++)
        {
            if (i == a[j]) c_count++;
        }
        c[i] = c_count;
    }
    //统计小于等于元素X的数组个数,即图(b)
    for (int i = 0; i < max - 1; i++)
    {
        c[i + 1] += c[i];

    }

    //开始排序,即图(c)(d)(e)
    for (int i = 0; i < count; i++)
    {
        b[c[a[i]] - 1] = a[i];
        c[a[i]]--;
    }
    return b;
}

void main()
{
    int count, *p;
    printf("请输入需要排序的数的个数 :");
    scanf_s("%d", &count);
    p = (int*)malloc(count * 2);
    printf("\n请输入需要排序的%d个数字:",count);
    for (int i = 0; i < count; i++)
    {
        scanf_s("%d", p+i);
    }
    printf("\n快速排序后的数组:");

    p = counting_sort(p, count);
    for (int i = 0; i < count; i++)
    {
        printf("%d ", p[i]);
    }
    printf("\n");
    system("pause");
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值