计数排序

计数排序假设n个输入元素中的每一个都是介于0-k的整数,此处k为某个整数。计数排序顾名思义离不开计数,我们要计的是输入元素中相同元素出现的次数。对每一个输入元素x,确定小于x的元素的个数,那样排序之后,x在最终输出数组中的位置就可以确定了。例如:如果有17个元素小于x,则x就位于第18个输出的位置上。当然有几个元素相同时,这个方法就略微改进一下,因为不能将它们放在同一个位置上。

假定输入数组为A[1..n],他们的值均位于0~k之间,输出排序之后的数组为B[1..n],以及临时存储数组C[0..k]。计数排序的伪代码如下:

伪代码代码   收藏代码
  1. memset(C,sizeof(C),0);       //C数组置零    
  2. for i=1 to n do    
  3.     C[A[i]]++;               //统计输入数组中相同元素的个数    
  4. for i=2 to k do    
  5.     C[i] = C[i]+C[i-1];       //C[i]表示输入数组中小于或者等于i的元素个数    
  6. for i=n downto 1 do    
  7.     B[C[A[i]]] = A[i];       //把每一个A[i]放到输出数组中相应位置上    
  8.     C[A[i]]--;             //如果有几个相同元素时,当然不能放在同一个位置了。  

 

计数排序特点:

1.  提前必须是已知待排序的关键字为整型且范围已知。

2.  时间复杂度为O(n+k),不是基于比较的排序算法,因此效率非常之高。

3.  稳定性好,这个是计数排序非常重要的特性,可以用在后面介绍的基数排序中。

4.  但需要一些辅助数组,如C[0..k],因此待排序的关键字范围0~k不宜过大。而B[1..n]用来存放排序结果,我们可以对上述算法进行改进,使排序在原地进行。改进之后如下:

伪代码代码   收藏代码
  1. memset(C,sizeof(C),0);       //C数组置零    
  2. for i=1 to n do    
  3.         C[A[i]]++;               //统计输入数组中相同元素的个数    
  4. idx = 0;    
  5. for i=0 to k do                 
  6.         while(C[i]>0) do         //C[i]中保存的是值为i元素的个数    
  7.             A[idx++] = i;        //因此很容易找到i在A中适合的位置    
  8.             C[i]--;     

 

最终实现的C代码如下:

C代码   收藏代码
  1. int countingSort(int *ar, int n, int k) {  
  2.     int i, idx = 0;  
  3.     int *B = calloc(k, sizeof(int));  
  4.   
  5.     for (i = 0; i < n; i++) {  
  6.         B[ar[i]]++;  
  7.     }  
  8.   
  9.     for (i = 0; i < k; i++) {  
  10.         while (B[i]-- > 0) {  
  11.             ar[idx++] = i;  
  12.         }  
  13.     }  
  14.     free(B);  
  15. }  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值