排序算法(一) 最快最简单的排序算法——桶排序

在生活中,到处都存在排序,比如考试排名、价格排序等

下面以成绩排序为例子学习桶排序算法

1、问题描述

现有5个同学,成绩分别是5分、3分、5分、2分、8分(满分10分),按照从大到小排序后是8 5 5 3 2,请用一段程序实现这个问题。

2、解决思路

首先我们需要申请一个大小为 11 的数组 int a[11]。OK ,现在你已经有了 11 个变量,编号从a[0]~a[10]。刚开始的时候,我们将 a[0]~a[10]都初始化为 0,表示这些分数还都没有人得过。例如 a[0]等于 0 就表示目前还没有人得过 0 分,同理 a[1]等于 0 就表示目前还没有人得过 1 分……a[10]等于 0 就表示目前还没有人得过 10 分。
3、过程讲解
下面开始处理每一个人的分数,第一个人的分数是 5 分,我们就将相对应 a[5]的值在原来的基础增加 1,即将 a[5]的值从 0 改为 1,表示 5 分出现过了一次。

第二个人的分数是 3 分,我们就把相对应 a[3]的值在原来的基础上增加 1,即将 a[3]的值从 0 改为 1,表示 3 分出现过了一次。

注意啦!第三个人的分数也是“5 分”,所以a[5]的值需要在此基础上再增加 1,即将 a[5]的值从 1 改为 2。表示 5 分出现过了两次。

 

依次类推, 最终结果就是下面这个图啦

你发现没有,a[0]~a[10]中的数值其实就是 0 分到 10 分每个分数出现的次数。接下来,我们只需要将出现过的分数打印出来就可以了,出现几次就打印几次,具体如下。   
       a[0]为 0,表示“0”没有出现过,不打印。
  a[1]为 0,表示“1”没有出现过,不打印。
  a[2]为 1,表示“2”出现过 1 次,打印 2。
  a[3]为 1,表示“3”出现过 1 次,打印 3。
  a[4]为 0,表示“4”没有出现过,不打印。
  a[5]为 2,表示“5”出现过 2 次,打印5 5。
     a[6]为 0,表示“6”没有出现过,不打印。
  a[7]为 0,表示“7”没有出现过,不打印。
  a[8]为 1,表示“8”出现过 1 次,打印 8。
  a[9]为 0,表示“9”没有出现过,不打印。
  a[10]为 0,表示“10”没有出现过,不打印。
  最终屏幕输出“2 3 5 5 8”,完整的代码如下。
        
#include <stdio.h>
int main()
{
  int a[11],i,j,t;
  for(i=0;i<=10;i++)
    a[i]=0; //初始化为0

  for(i=1;i<=5;i++) //循环读入5个数
  {
    scanf("%d",&t); //把每一个数读到变量t中
    a[t]++; //进行计数
  }

  for(i=0;i<=10;i++) //依次判断a[0]~a[10]
    for(j=1;j<=a[i];j++) //出现了几次就打印几次
      printf("%d ",i);

  getchar();
  getchar();
  //这里的getchar();用来暂停程序,以便查看程序输出的内容
  //也可以用system("pause");等来代替
  return 0;
}
最后来说下时间复杂度的问题。首先,初始赋值 0 共执行n次(n个桶)。接着,桶中数据计数一共循环了 m 次(m 个数据),一共循环了 m+n 次。所以整个排序算法一共执行了 m+n次。最后打印执行了m+n次,所以该程序总计 2(m+n)次。
我们用大写字母 O 来表示时间复杂度,因此该算法的时间复杂度是 O( 2*(m+n) )。我们在说时间复杂度时候可以忽略较小的常数,最终桶排序的时间复杂度为 O(m+n)。还有一点,在表示时间复杂度的时候,n 和 m 通常用大写字母即 O(M+N)
这是一个非常快的排序算法。桶排序从 1956 年就开始被使用,该算法的基本思想是由 E.J.Issac R.C.Singleton 提出来。之前说过,其实这并不是真正的桶排序算法,真正的桶排序算法要比这个更加复杂。但是考虑到此处是算法讲解的第一篇,我想还是越简单易懂越好,真正的桶排序留在以后再聊吧。需要说明一点的是:我们目前学习的简化版桶排序算法其本质上还不能算是一个真正意义上的排序算法。为什么呢?例如遇到下面这个例子就没辙了。
现在分别有 5 个人的名字和分数:huhu 5 分、haha 3 分、xixi 5 分、hengheng 2 分和 gaoshou 8 分。请按照分数从高到低,输出他们的名字。即应该输出 gaoshou、huhu、xixi、haha、hengheng。
发现问题了没有?如果使用我们刚才简化版的桶排序算法仅仅是把分数进行了排序。最终输出的也仅仅是分数,但没有对人本身进行排序。也就是说,我们现在并不知道排序后的分数原本对应着哪一个人!这该怎么办呢?不要着急请听下回—— 冒泡排序
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值