【初阶数据结构与算法】八大排序之非比较排序(计数排序),一次性讲清!

(一)计数排序 Countsort

(升序为例)开辟一个额外数组count用于存储待排数组中的元素个数,譬如待排数组为{2,2,3,3,5,5,5,6,3,1},这样1的个数有1个,存在count数组的第一位中,2有2个,存在count数组的第二位中,3有3个,存在count数组的第三位中,4有0个,存在count数组的第四位中,5有3个,存在count数组的第五位中,6有1个,存在count数组的第六位中。

即arr中的数据是啥,就将它的个数存储到count对应下标的位置处。

然后再根据count数组中存储的数据个数,依次将值拷贝回原数组中,1有1个,放回原数组中,占一个位置,2有2个,放回原数组中,挨着1占两个位置,3有3个,放回原数组中,挨着2占三个位置,4放回原数组中,挨着3占零个位置,5有3个,放回原数组中,挨着4占三个位置,6有一个,放回原数组中,挨着5占一个位置。

排好序后,原数组变成了{1,2,2,3,3,3,5,5,5,6},排升序完成。

(1)代码实现

复制代码
void Countsort(int* arr,int n)
{
  int* tmp = (int*)calloc(n,sizeof(int));
  if(tmp == NULL)
  {
      return;
  }
  int* count = tmp;
  //找最大最小值
  int max = arr[0];
  int min = arr[0];
  for(int i=1;i<n;i++)
  {
    if(arr[i]>max)
       max = arr[i];
    if(arr[i]<min)
       min = arr[i];
  }
  int range = max-min+1;
  //定义数组的长度
  for(int i = 0;i<n;i++)
  {
  //让count数组相应下标处存储处理后的arr数组的数据的出现次数
    count[arr[i]-min]++;
  }
  //让count数组中存储的个数显化为arr数组的排序
  int j = 0;
  for(int i = 0;i<range;i++)
  {
    while(count[i]--)
       arr[j++] = i+min;
  }

}

(2)细节理解

数组里存的数据不同,count数组的大小就不同,所以我们采用动态申请内存的方式创建数组。

要确定count数组的大小,就要知道arr中的数据范围是多少,因为count数组存储的是arr数组中各个数据的出现次数,找到arr数组数据的取值范围,就知道count数组需要记录多少个数据的出现次数,这个"多少个数据"就是count数组的大小。

所以我们找到arr数组中的max和min,通过max-min+1的方式来确定count数组的大小range。

接下来,遍历arr数组,前面我们说arr中的数据是啥,就将它的个数存储到count对应下标的位置处,但这句话是有问题的,假设数组arr为{103,102,109,105},我们能确定count数组的大小为109-102+1=8,那count数组的下标范围就为0-7,何来103/102/109/105呢?

所以,arr中数据要先处理一下,再看这个处理后的数据,处理后的数据是啥,就将它的个数存储到count数组对应下标的位置。

处理的方式就是将arr数组中的数据先减去min。

arr为{103,102,109,105},处理后就变成了{1,0,7,3},这样count数组0-7的下标就能满足存储要求了,由于0-7范围内,arr只出现了1,0,7,3,还有一些数没有出现,但是count数组中已经为这些没有出现的数据创建了空间,所以这些空间里填的值应当为0,这也对应了为什么我们用calloc动态申请内存,calloc申请空间,并将空间全部初始化为0。

等到将count数组中存储的数据个数显化成arr中的数据时,遍历count数组,内嵌循环赋值,注意,由于count数组存储的数据个数,是处理后的数据的个数,那么再赋值回arr数组时,就需要将count的下标加上min。

(二)基数排序 Radixsort

基数排序(Radix Sort)是一种非比较型的排序算法,它通过逐位比较元素的每一位(从最低位到最高位)来实现排序。基数排序的核心思想是将整数按位数切割成不同的数字,然后按每个位数分别进行排序。

基数排序算法适用于对多个整数或者多个字符串进行升序或降序排序。

(三)桶排序 Bucketsort

桶排序(Bucket Sort)是一种分布式排序算法,它将待排序的元素分配到若干个桶(Bucket)中,然后对每个桶中的元素进行排序,最后将所有桶中的元素按顺序合并。

桶排序的核心思想是将数据分到有限数量的桶中,每个桶再分别排序(可以使用其他排序算法或递归地使用桶排序)。

后两者出现频率不高,此处不过多解释了。

------end------

相关推荐
罗西的思考2 小时前
【Agentic RL / 强化学习 / OPD】OpenClaw-RL 源码阅读笔记 --- (4)--- 系统架构
人工智能·算法·机器学习
QiLinkOS2 小时前
从技术到资产的跃迁:企业专利布局的深层逻辑
c语言·数据结构·c++·单片机·嵌入式硬件·算法·开源
aini_lovee2 小时前
FMCW雷达测速测距系统(锯齿波 + CFAR检测)
算法
qq_297574672 小时前
设计模式系列文章(基础篇第 11 篇):模板方法模式——定义算法骨架,实现代码复用与流程统一
算法·设计模式·模板方法模式
lqqjuly2 小时前
知识蒸馏:理论、算法与可运行实现
人工智能·深度学习·算法
水上冰石2 小时前
comfui的sd1.5模型,有多少采样算法,详解每一个采样算法
人工智能·算法
黎阳之光3 小时前
视频孪生+空天地水工融合,黎阳之光构建智慧水利监测新范式
大数据·人工智能·物联网·算法·安全
cheems95273 小时前
[算法手记] 贪心 爬楼梯问题
算法·贪心算法
KaMeidebaby3 小时前
卡梅德生物技术快报|酵母双杂交 cDNA 文库构建与蛋白互作筛选流程
服务器·前端·数据库·人工智能·算法