传说时间复杂度为0(n)的排序

2024-03-18 04:18
文章标签 复杂度 时间 排序 传说

本文主要是介绍传说时间复杂度为0(n)的排序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

线性时间的排序算法

   前面已经介绍了几种排序算法,像插入排序(直接插入排序,折半插入排序,希尔排序)、交换排序(冒泡排序,快速排序)、选择排序(简单选择排序,堆排序)、2-路归并排序(见我的另一篇文章:各种内部排序算法的实现)等,这些排序算法都有一个共同的特点,就是基于比较。本文将介绍三种非比较的排序算法:计数排序,基数排序,桶排序。它们将突破比较排序的Ω(nlgn)下界,以线性时间运行。

一、比较排序算法的时间下界

所谓的比较排序是指通过比较来决定元素间的相对次序。

“定理:对于含n个元素的一个输入序列,任何比较排序算法在最坏情况下,都需要做Ω(nlgn)次比较。

也就是说,比较排序算法的运行速度不会快于nlgn,这就是基于比较的排序算法的时间下界

通过决策树(Decision-Tree)可以证明这个定理,关于决策树的定义以及证明过程在这里就不赘述了。你可以自己去查找资料,推荐观看《MIT公开课:线性时间排序》。

根据上面的定理,我们知道任何比较排序算法的运行时间不会快于nlgn。那么我们是否可以突破这个限制呢?当然可以,接下来我们将介绍三种线性时间的排序算法,它们都不是通过比较来排序的,因此,下界Ω(nlgn)对它们不适用。

二、计数排序(Counting Sort)

计数排序的基本思想就是对每一个输入元素x,确定小于x的元素的个数,这样就可以把x直接放在它在最终输出数组的位置上,例如:

算法的步骤大致如下:

  • 找出待排序的数组中最大和最小的元素

  • 统计数组中每个值为i的元素出现的次数,存入数组C的第i项

  • 对所有的计数累加(从C中的第一个元素开始,每一项和前一项相加)

  • 反向填充目标数组:将每个元素i放在新数组的第C(i)项,每放一个元素就将C(i)减去1

C++代码

  1. /************************************************************************* 
  2.     > File Name: CountingSort.cpp 
  3.     > Author: SongLee 
  4.     > E-mail: lisong.shine@qq.com 
  5.     > Created Time: 2014年06月11日 星期三 00时08分55秒 
  6.     > Personal Blog: http://songlee24.github.io 
  7.  ************************************************************************/  
  8. #include<iostream>  
  9. using namespace std;  
  10.   
  11. /* 
  12.  *计数排序:A和B为待排和目标数组,k为数组中最大值,len为数组长度 
  13.  */  
  14. void CountingSort(int A[], int B[], int k, int len)  
  15. {  
  16.     int C[k+1];  
  17.     for(int i=0; i<k+1; ++i)  
  18.         C[i] = 0;  
  19.     for(int i=0; i<len; ++i)  
  20.         C[A[i]] += 1;  
  21.     for(int i=1; i<k+1; ++i)  
  22.         C[i] = C[i] + C[i-1];  
  23.     for(int i=len-1; i>=0; --i)  
  24.     {  
  25.         B[C[A[i]]-1] = A[i];  
  26.         C[A[i]] -= 1;  
  27.     }  
  28. }  
  29.   
  30. /* 输出数组 */  
  31. void print(int arr[], int len)  
  32. {  
  33.     for(int i=0; i<len; ++i)  
  34.         cout << arr[i] << " ";  
  35.     cout << endl;  
  36. }  
  37.   
  38. /* 测试 */  
  39. int main()  
  40. {  
  41.     int origin[8] = {4,5,3,0,2,1,15,6};  
  42.     int result[8];  
  43.     print(origin, 8);  
  44.     CountingSort(origin, result, 15, 8);  
  45.     print(result, 8);  
  46.     return 0;  
  47. }  
当输入的元素是0到k之间的整数时,时间复杂度是O(n+k),空间复杂度也是O(n+k)。当k不是很大并且序列比较集中时,计数排序是一个很有效的排序算法。计数排序是一个稳定的排序算法。

可能你会发现,计数排序似乎饶了点弯子,比如当我们刚刚统计出C,C[i]可以表示A中值为i的元素的个数,此时我们直接顺序地扫描C,就可以求出排序后的结果。的确是这样,不过这种方法不再是计数排序,而是桶排序,确切地说,是桶排序的一种特殊情况。

三、桶排序(Bucket Sort)

桶排序(Bucket Sort)的思想是将数组分到有限数量的桶子里。每个桶子再个别排序(有可能再使用别的排序算法)。当要被排序的数组内的数值是均匀分配的时候,桶排序可以以线性时间运行。桶排序过程动画演示:Bucket Sort,桶排序原理图如下:

C++代码

  1. /************************************************************************* 
  2.     > File Name: BucketSort.cpp 
  3.     > Author: SongLee 
  4.     > E-mail: lisong.shine@qq.com 
  5.     > Created Time: 2014年06月11日 星期三 09时17分32秒 
  6.     > Personal Blog: http://songlee24.github.io 
  7.  ************************************************************************/  
  8. #include<iostream>  
  9. using namespace std;  
  10.   
  11. /* 节点 */  
  12. struct node  
  13. {  
  14.     int value;  
  15.     node* next;  
  16. };  
  17.   
  18. /* 桶排序 */  
  19. void BucketSort(int A[], int max, int len)  
  20. {  
  21.     node bucket[len];  
  22.     int count=0;  
  23.     for(int i=0; i<len; ++i)  
  24.     {  
  25.         bucket[i].value = 0;  
  26.         bucket[i].next = NULL;  
  27.     }  
  28.       
  29.     for(int i=0; i<len; ++i)  
  30.     {  
  31.         node *ist = new node();  
  32.         ist->value = A[i];  
  33.         ist->next = NULL;  
  34.         int idx = A[i]*len/(max+1); // 计算索引  
  35.         if(bucket[idx].next == NULL)  
  36.         {  
  37.             bucket[idx].next = ist;  
  38.         }  
  39.         else /* 按大小顺序插入链表相应位置 */  
  40.         {  
  41.             node *p = &bucket[idx];  
  42.             node *q = p->next;  
  43.             while(q!=NULL && q->value <= A[i])  
  44.             {  
  45.                 p = q;  
  46.                 q = p->next;  
  47.             }  
  48.             ist->next = q;  
  49.             p->next = ist;  
  50.         }  
  51.     }  
  52.   
  53.     for(int i=0; i<len; ++i)  
  54.     {  
  55.         node *p = bucket[i].next;  
  56.         if(p == NULL)  
  57.             continue;  
  58.         while(p!= NULL)  
  59.         {  
  60.             A[count++] = p->value;  
  61.             p = p->next;  
  62.         }  
  63.     }  
  64. }  
  65.   
  66. /* 输出数组 */  
  67. void print(int A[], int len)  
  68. {  
  69.     for(int i=0; i<len; ++i)  
  70.         cout << A[i] << " ";  
  71.     cout << endl;  
  72. }  
  73.   
  74. /* 测试 */  
  75. int main()  
  76. {  
  77.     int row[11] = {24,37,44,12,89,93,77,61,58,3,100};  
  78.     print(row, 11);  
  79.     BucketSort(row, 235, 11);  
  80.     print(row, 11);  
  81.     return 0;  
  82. }  

四、基数排序(Radix Sort)

基数排序(Radix Sort)是一种非比较型排序算法,它将整数按位数切割成不同的数字,然后按每个位分别进行排序。基数排序的方式可以采用MSD(Most significant digital)或LSD(Least significant digital),MSD是从最高有效位开始排序,而LSD是从最低有效位开始排序。

当然我们可以采用MSD方式排序,按最高有效位进行排序,将最高有效位相同的放到一堆,然后再按下一个有效位对每个堆中的数递归地排序,最后再将结果合并起来。但是,这样会产生很多中间堆。所以,通常基数排序采用的是LSD方式。

LSD基数排序实现的基本思路是将所有待比较数值(正整数)统一为同样的数位长度,数位较短的数前面补零。然后,从最低位开始,依次进行一次排序。这样从最低位排序一直到最高位排序完成以后, 数列就变成一个有序序列。需要注意的是,对每一个数位进行排序的算法必须是稳定的,否则就会取消前一次排序的结果。通常我们使用计数排序或者桶排序作为基数排序的辅助算法。基数排序过程动画演示:Radix Sort

C++实现(使用计数排序)

  1. /************************************************************************* 
  2.     > File Name: RadixSort.cpp 
  3.     > Author: SongLee 
  4.     > E-mail: lisong.shine@qq.com 
  5.     > Created Time: 2014年06月22日 星期日 12时04分37秒 
  6.     > Personal Blog: http://songlee24.github.io 
  7.  ************************************************************************/  
  8. #include<iostream>  
  9. using namespace std;  
  10.   
  11. // 找出整数num第n位的数字  
  12. int findIt(int num, int n)  
  13. {  
  14.     int power = 1;  
  15.     for (int i = 0; i < n; i++)  
  16.     {  
  17.         power *= 10;  
  18.     }  
  19.     return (num % power) * 10 / power;  
  20. }  
  21.   
  22. // 基数排序(使用计数排序作为辅助)  
  23. void RadixSort(int A[], int len, int k)  
  24. {  
  25.     for(int i=1; i<=k; ++i)  
  26.     {  
  27.         int C[10] = {0};   // 计数数组  
  28.         int B[len];        // 结果数组  
  29.   
  30.         for(int j=0; j<len; ++j)  
  31.         {  
  32.             int d = findIt(A[j], i);  
  33.             C[d] += 1;  
  34.         }  
  35.   
  36.         for(int j=1; j<10; ++j)  
  37.             C[j] = C[j] + C[j-1];  
  38.   
  39.         for(int j=len-1; j>=0; --j)  
  40.         {  
  41.             int d = findIt(A[j], i);  
  42.             C[d] -= 1;  
  43.             B[C[d]] = A[j];  
  44.         }  
  45.           
  46.         // 将B中排好序的拷贝到A中  
  47.         for(int j=0; j<len; ++j)  
  48.             A[j] = B[j];  
  49.     }  
  50. }  
  51.   
  52. // 输出数组  
  53. void print(int A[], int len)  
  54. {  
  55.     for(int i=0; i<len; ++i)  
  56.         cout << A[i] << " ";  
  57.     cout << endl;  
  58. }  
  59.   
  60. // 测试  
  61. int main()  
  62. {  
  63.     int A[8] = {332, 653, 632, 5, 755, 433, 722, 48};  
  64.     print(A, 8);  
  65.     RadixSort(A, 8, 3);  
  66.     print(A, 8);  
  67.     return 0;  
  68. }  
基数排序的时间复杂度是 O(k·n),其中n是排序元素个数,k是数字位数。注意这不是说这个时间复杂度一定优于O(nlgn),因为n可能具有比较大的系数k。

另外,基数排序不仅可以对整数排序,也可以对有多个关键字域的记录进行排序。例如,根据三个关键字年、月、日来对日期进行排序。


这篇关于传说时间复杂度为0(n)的排序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/821148

相关文章

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

java中查看函数运行时间和cpu运行时间

android开发调查性能问题中有一个现象,函数的运行时间远低于cpu执行时间,因为函数运行期间线程可能包含等待操作。native层可以查看实际的cpu执行时间和函数执行时间。在java中如何实现? 借助AI得到了答案 import java.lang.management.ManagementFactory;import java.lang.management.Threa

时间服务器中,适用于国内的 NTP 服务器地址,可用于时间同步或 Android 加速 GPS 定位

NTP 是什么?   NTP 是网络时间协议(Network Time Protocol),它用来同步网络设备【如计算机、手机】的时间的协议。 NTP 实现什么目的?   目的很简单,就是为了提供准确时间。因为我们的手表、设备等,经常会时间跑着跑着就有误差,或快或慢的少几秒,时间长了甚至误差过分钟。 NTP 服务器列表 最常见、熟知的就是 www.pool.ntp.org/zo

20170723 做的事 ecdsa的签名验证时间短于bls signature

1 今天在虚拟机 /home/smile/Desktop/20170610/Test//time_ecdsa 文件夹下,找到ecdsa的验证时间是 989.060606μs μs 先 make ,然后run。 再取BLS的签名生成时间: ./run  2  gnuplot 画图,画对比的时间 gnuplot 画图参考教程 http://blog.sciencen

数据结构9——排序

一、冒泡排序 冒泡排序(Bubble Sort),顾名思义,就是指越小的元素会经由交换慢慢“浮”到数列的顶端。 算法原理 从左到右,依次比较相邻的元素大小,更大的元素交换到右边;从第一组相邻元素比较到最后一组相邻元素,这一步结束最后一个元素必然是参与比较的元素中最大的元素;按照大的居右原则,重新从左到后比较,前一轮中得到的最后一个元素不参4与比较,得出新一轮的最大元素;按照上述规则,每一轮结

七种排序方式总结

/*2018.01.23*A:YUAN*T:其中排序算法:冒泡排序,简单排序,直接插入排序,希尔排序,堆排序,归并排序,快速排序*/#include <stdio.h>#include <math.h>#include <malloc.h>#define MAXSIZE 10000#define FALSE 0#define TRUE 1typedef struct {i

Python几种建表方法运行时间的比较

建立一个表[0,1,2,3.......10n],下面几种方法都能实现,但是运行时间却截然不同哦 import time#方法一def test1(n):list=[]for i in range(n*10):list=list+[i]return list#方法二def test2(n):list=[]for i in range(n*10):list.append(i)#方法三d

拓扑排序——C语言

拓扑排序(Topological Sorting)是一种用于有向无环图(DAG)的排序算法,其输出是图中所有顶点的线性排序,使得对于每条有向边 (u, v),顶点 u 在 v 之前出现。拓扑排序确定了项目网络图中的起始事件和终止事件,也就是顶点的执行顺序。         因为是有向无环图,所以拓扑排序的作用其实就是把先发生的排序在前面,后发生的排序到后面。 例如现在我们有一个

【Qt6.3 基础教程 16】 掌握Qt中的时间和日期:QTimer和QDateTime的高效应用

文章目录 前言QTimer:定时任务的强大工具QTimer的基本用法高级特性:单次定时器 QDateTime:处理日期和时间获取当前日期和时间日期和时间的格式化输出日期和时间计算 用例:创建一个倒计时应用结论 前言 在开发桌面应用程序时,处理时间和日期是一个常见且重要的任务。Qt框架提供了强大的工具来处理与时间相关的功能,其中QTimer和QDateTime是最核心的类。本