码迷,mamicode.com
首页 > 编程语言 > 详细

C#排序算法的比较

时间:2014-11-06 14:38:58      阅读:305      评论:0      收藏:0      [点我收藏+]

标签:style   http   io   color   ar   for   sp   strong   div   

首先通过图表比较不同排序算法的时间复杂度和稳定性。

 

排序方法

平均时间

最坏情况

最好情况

辅助空间

稳定性

直接插入排序

O(n2)

O(n2)

O(n)

O(1)

冒泡排序

O(n2)

O(n2)

O(n)

O(1)

简单选择排序

O(n2)

O(n2)

O(n2)

O(1)

希尔排序 -

O(nlog2n)~O(n2)

O(nlog2n)~O(n2)

O(1)

快速排序

O(nlog2n)

O(n2)

O(nlog2n)

O(log2n)

堆排序

O(nlog2n)

O(nlog2n)

O(nlog2n)

O(1)

2-路归并排序

O(nlog2n)

O(nlog2n)

O(nlog2n)

O(n)

基数排序 O(d(n + rd)) O(d(n + rd)) O(d(n + rd)) O(rd)

注:1. 算法的时间复杂度一般情况下指最坏情况下的渐近时间复杂度。

        2. 排序算法的稳定性会对多关键字排序产生影响。

 

下面通过C#代码说明不同的排序算法

 

插入排序

时间复杂度:平均情况—O(n2) 最坏情况—O(n2) 辅助空间:O(1) 稳定性:稳定

插入排序是在一个已经有序的小序列的基础上,一次插入一个元素。当然,刚开始这个有序的小序列只有1个元素,就是第一个元素。比较是从有序序列的末尾开始,也就是想要插入的元素和已经有序的最大者开始比起,如果比它大则直接插入在其后面,否则一直往前找直到找到它该插入的位置。如果碰见一个和插入元素相等的,那么插入元素把想插入的元素放在相等元素的后面。所以,相等元素的前后顺序没有改变,从原无序序列出去的顺序就是排好序后的顺序,所以插入排序是稳定的。

 
C# 代码   复制
bubuko.com,布布扣
bubuko.com,布布扣void InsertSort(SqList &L) {
bubuko.com,布布扣
bubuko.com,布布扣  // 对顺序表L作直接插入排序。
bubuko.com,布布扣
bubuko.com,布布扣  int i,j;
bubuko.com,布布扣
bubuko.com,布布扣  for (i=2; i<=L.length; ++i)
bubuko.com,布布扣
bubuko.com,布布扣    if (LT(L.r[i].key, L.r[i-1].key)) {
bubuko.com,布布扣
bubuko.com,布布扣      // "<"时,需将L.r[i]插入有序子表
bubuko.com,布布扣
bubuko.com,布布扣      L.r[0] = L.r[i];                 // 复制为哨兵
bubuko.com,布布扣
bubuko.com,布布扣      for (j=i-1;  LT(L.r[0].key, L.r[j].key);  --j)
bubuko.com,布布扣
bubuko.com,布布扣        L.r[j+1] = L.r[j];             // 记录后移
bubuko.com,布布扣
bubuko.com,布布扣      L.r[j+1] = L.r[0];               // 插入到正确位置
bubuko.com,布布扣
bubuko.com,布布扣    }
bubuko.com,布布扣
bubuko.com,布布扣} // InsertSort   

 

希尔排序(shell)

时间复杂度:理想情况—O(nlog2n) 最坏情况—O(n2) 稳定性:不稳定


希尔排序是按照不同步长对元素进行插入排序,当刚开始元素很无序的时候,步长最大,所以插入排序的元素个数很少,速度很快;当元素基本有序了,步长很小,插入排序对于有序的序列效率很高。所以,希尔排序的时间复杂度会比o(n^2)好一些。由于多次插入排序,我们知道一次插入排序是稳定的,不会改变相同元素的相对顺序,但在不同的插入排序过程中,相同的元素可能在各自的插入排序中移动,最后其稳定性就会被打乱,所以shell排序是不稳定的。

 
C# 代码   复制
bubuko.com,布布扣
bubuko.com,布布扣void ShellInsert(SqList &L, int dk) {
bubuko.com,布布扣
bubuko.com,布布扣  // 对顺序表L作一趟希尔插入排序。本算法对算法10.1作了以下修改:
bubuko.com,布布扣
bubuko.com,布布扣  //     1. 前后记录位置的增量是dk,而不是1;
bubuko.com,布布扣
bubuko.com,布布扣  //     2. r[0]只是暂存单元,不是哨兵。当j<=0时,插入位置已找到。
bubuko.com,布布扣
bubuko.com,布布扣  int i,j;
bubuko.com,布布扣
bubuko.com,布布扣  for (i=dk+1; i<=L.length; ++i)
bubuko.com,布布扣
bubuko.com,布布扣    if (LT(L.r[i].key, L.r[i-dk].key)) { // 需将L.r[i]插入有序增量子表
bubuko.com,布布扣
bubuko.com,布布扣      L.r[0] = L.r[i];                   // 暂存在L.r[0]
bubuko.com,布布扣
bubuko.com,布布扣      for (j=i-dk; j>0 && LT(L.r[0].key, L.r[j].key); j-=dk)
bubuko.com,布布扣
bubuko.com,布布扣        L.r[j+dk] = L.r[j];              // 记录后移,查找插入位置
bubuko.com,布布扣
bubuko.com,布布扣      L.r[j+dk] = L.r[0];                // 插入
bubuko.com,布布扣
bubuko.com,布布扣    }
bubuko.com,布布扣
bubuko.com,布布扣} // ShellInsert  
bubuko.com,布布扣
bubuko.com,布布扣  
bubuko.com,布布扣
bubuko.com,布布扣void ShellSort(SqList &L, int dlta[], int t) {
bubuko.com,布布扣
bubuko.com,布布扣   // 按增量序列dlta[0..t-1]对顺序表L作希尔排序。
bubuko.com,布布扣
bubuko.com,布布扣   for (int k=0;k<t;k++)
bubuko.com,布布扣
bubuko.com,布布扣      ShellInsert(L, dlta[k]);  // 一趟增量为dlta[k]的插入排序
bubuko.com,布布扣
bubuko.com,布布扣} // ShellSort  

 

冒泡排序

时间复杂度:平均情况—O(n2) 最坏情况—O(n2) 辅助空间:O(1) 稳定性:稳定

冒泡排序就是把小的元素往前调或者把大的元素往后调。比较是相邻的两个元素比较,交换也发生在这两个元素之间。所以,如果两个元素相等,我想你是不会再无聊地把他们俩交换一下的;如果两个相等的元素没有相邻,那么即使通过前面的两两交换把两个相邻起来,这时候也不会交换,所以相同元素的前后顺序并没有改变,所以冒泡排序是一种稳定排序算法。

 
C# 代码   复制
bubuko.com,布布扣
bubuko.com,布布扣void BubbleSort(SeqList R) {
bubuko.com,布布扣
bubuko.com,布布扣  int i,j;
bubuko.com,布布扣
bubuko.com,布布扣  Boolean exchange; //交换标志
bubuko.com,布布扣
bubuko.com,布布扣  for(i=1;i<n;i++){ exchange="FALSE;" j="n-1;j">=i;j--) //对当前无序区R[i..n]自下向上扫描
bubuko.com,布布扣
bubuko.com,布布扣            if(R[j+1].key< R[j].key){//交换记录
bubuko.com,布布扣
bubuko.com,布布扣                R[0]=R[j+1]; //R[0]不是哨兵,仅做暂存单元
bubuko.com,布布扣
bubuko.com,布布扣                R[j+1]=R[j];
bubuko.com,布布扣
bubuko.com,布布扣                R[j]=R[0];
bubuko.com,布布扣
bubuko.com,布布扣                exchange=TRUE; //发生了交换,故将交换标志置为真
bubuko.com,布布扣
bubuko.com,布布扣            }
bubuko.com,布布扣
bubuko.com,布布扣            if(!exchange) //本趟排序未发生交换,提前终止算法
bubuko.com,布布扣
bubuko.com,布布扣            return;
bubuko.com,布布扣
bubuko.com,布布扣  } //endfor(外循环)
bubuko.com,布布扣
bubuko.com,布布扣}

 

快速排序

时间复杂度:平均情况—O(nlog2n) 最坏情况—O(n2) 辅助空间:O(log2n) 稳定性:不稳定

快速排序有两个方向,左边的i下标一直往右走,当a[i] <= a[center_index],其中center_index是中枢元素的数组下标,一般取为数组第0个元素。而右边的j下标一直往左走,当a[j] > a[center_index]。如果i和j都走不动了,i <= j, 交换a[i]和a[j],重复上面的过程,直到i>j。 交换a[j]和a[center_index],完成一趟快速排序。在中枢元素和a[j]交换的时候,很有可能把前面的元素的稳定性打乱,比如序列为 5 3 3 4 3 8 9 10 11, 现在中枢元素5和3(第5个元素,下标从1开始计)交换就会把元素3的稳定性打乱,所以快速排序是一个不稳定的排序算法,不稳定发生在中枢元素和a[j]交换的时刻。

 
C# 代码   复制
bubuko.com,布布扣
bubuko.com,布布扣int Partition(SqList &L, int low, int high) {
bubuko.com,布布扣
bubuko.com,布布扣 // 交换顺序表L中子序列L.r[low..high]的记录,使枢轴记录到位,
bubuko.com,布布扣
bubuko.com,布布扣   // 并返回其所在位置,此时,在它之前(后)的记录均不大(小)于它
bubuko.com,布布扣
bubuko.com,布布扣   KeyType pivotkey;
bubuko.com,布布扣
bubuko.com,布布扣   RedType temp;
bubuko.com,布布扣
bubuko.com,布布扣   pivotkey = L.r[low].key;     // 用子表的第一个记录作枢轴记录
bubuko.com,布布扣
bubuko.com,布布扣   while (low < high) {           // 从表的两端交替地向中间扫描
bubuko.com,布布扣
bubuko.com,布布扣      while (low < high && L.r[high].key>=pivotkey) --high;
bubuko.com,布布扣
bubuko.com,布布扣      temp=L.r[low];
bubuko.com,布布扣
bubuko.com,布布扣      L.r[low]=L.r[high];
bubuko.com,布布扣
bubuko.com,布布扣      L.r[high]=temp;           // 将比枢轴记录小的记录交换到低端
bubuko.com,布布扣
bubuko.com,布布扣      while (low  < high && L.r[low].key < =pivotkey) ++low;
bubuko.com,布布扣
bubuko.com,布布扣      temp=L.r[low];
bubuko.com,布布扣
bubuko.com,布布扣      L.r[low]=L.r[high];
bubuko.com,布布扣
bubuko.com,布布扣      L.r[high]=temp;           // 将比枢轴记录大的记录交换到高端
bubuko.com,布布扣
bubuko.com,布布扣   }
bubuko.com,布布扣
bubuko.com,布布扣   return low;                  // 返回枢轴所在位置
bubuko.com,布布扣
bubuko.com,布布扣} // Partition        
bubuko.com,布布扣
bubuko.com,布布扣
bubuko.com,布布扣void QSort(SqList &L, int low, int high) {
bubuko.com,布布扣
bubuko.com,布布扣  // 对顺序表L中的子序列L.r[low..high]进行快速排序
bubuko.com,布布扣
bubuko.com,布布扣  int pivotloc;
bubuko.com,布布扣
bubuko.com,布布扣  if (low  <  high) {                      // 长度大于1
bubuko.com,布布扣
bubuko.com,布布扣    pivotloc = Partition(L, low, high);  // 将L.r[low..high]一分为二
bubuko.com,布布扣
bubuko.com,布布扣    QSort(L, low, pivotloc-1); // 对低子表递归排序,pivotloc是枢轴位置
bubuko.com,布布扣
bubuko.com,布布扣    QSort(L, pivotloc+1, high);          // 对高子表递归排序
bubuko.com,布布扣
bubuko.com,布布扣  }
bubuko.com,布布扣
bubuko.com,布布扣} // QSort     
bubuko.com,布布扣
bubuko.com,布布扣  
bubuko.com,布布扣
bubuko.com,布布扣void QuickSort(SqList &L) {
bubuko.com,布布扣
bubuko.com,布布扣   // 对顺序表L进行快速排序
bubuko.com,布布扣
bubuko.com,布布扣   QSort(L, 1, L.length);
bubuko.com,布布扣
bubuko.com,布布扣} // QuickSort 

 

选择排序

时间复杂度:平均情况—O(n2) 最坏情况—O(n2) 辅助空间:O(1) 稳定性:不稳定

选择排序是给每个位置选择当前元素最小的,比如给第一个位置选择最小的,在剩余元素里面给第二个元素选择第二小的,依次类推,直到第n-1个元素,第n个元素不用选择了,因为只剩下它一个最大的元素了。那么,在一趟选择,如果当前元素比一个元素小,而该小的元素又出现在一个和当前元素相等的元素后面,那么交换后稳定性就被破坏了。比较拗口,举个例子,序列5 8 5 2 9, 我们知道第一遍选择第1个元素5会和2交换,那么原序列中2个5的相对前后顺序就被破坏了,所以选择排序不是一个稳定的排序算法。

 
C# 代码   复制
bubuko.com,布布扣
bubuko.com,布布扣void SelectSort(SqList &L) {
bubuko.com,布布扣
bubuko.com,布布扣  // 对顺序表L作简单选择排序。
bubuko.com,布布扣
bubuko.com,布布扣  int i,j;
bubuko.com,布布扣
bubuko.com,布布扣  for (i=1; i < L.length; ++i) { // 选择第i小的记录,并交换到位
bubuko.com,布布扣
bubuko.com,布布扣    j = SelectMinKey(L, i);  // 在L.r[i..L.length]中选择key最小的记录
bubuko.com,布布扣
bubuko.com,布布扣    if (i!=j) {                // L.r[i]←→L.r[j];   与第i个记录交换
bubuko.com,布布扣
bubuko.com,布布扣      RedType temp;
bubuko.com,布布扣
bubuko.com,布布扣      temp=L.r[i];
bubuko.com,布布扣
bubuko.com,布布扣      L.r[i]=L.r[j];
bubuko.com,布布扣
bubuko.com,布布扣      L.r[j]=temp;
bubuko.com,布布扣
bubuko.com,布布扣    }
bubuko.com,布布扣
bubuko.com,布布扣  }
bubuko.com,布布扣
bubuko.com,布布扣} // SelectSort    

 

堆排序

时间复杂度:平均情况—O(nlog2n) 最坏情况—O(nlog2n) 辅助空间:O(1) 稳定性:不稳定


我们知道堆的结构是节点i的孩子为2*i和2*i+1节点,大顶堆要求父节点大于等于其2个子节点,小顶堆要求父节点小于等于其2个子节点。在一个长为n的序列,堆排序的过程是从第n/2开始和其子节点共3个值选择最大(大顶堆)或者最小(小顶堆),这3个元素之间的选择当然不会破坏稳定性。但当为n/2-1, n/2-2, ...1这些个父节点选择元素时,就会破坏稳定性。有可能第n/2个父节点交换把后面一个元素交换过去了,而第n/2-1个父节点把后面一个相同的元素没有交换,那么这2个相同的元素之间的稳定性就被破坏了。所以,堆排序不是稳定的排序算法

 
C# 代码   复制
bubuko.com,布布扣
bubuko.com,布布扣void HeapAdjust(HeapType &H, int s, int m) {
bubuko.com,布布扣
bubuko.com,布布扣  // 已知H.r[s..m]中记录的关键字除H.r[s].key之外均满足堆的定义,
bubuko.com,布布扣
bubuko.com,布布扣  // 本函数调整H.r[s]的关键字,使H.r[s..m]成为一个大顶堆
bubuko.com,布布扣
bubuko.com,布布扣  // (对其中记录的关键字而言)
bubuko.com,布布扣
bubuko.com,布布扣  int j;
bubuko.com,布布扣
bubuko.com,布布扣  RedType rc;
bubuko.com,布布扣
bubuko.com,布布扣  rc = H.r[s];
bubuko.com,布布扣
bubuko.com,布布扣  for (j=2*s; j < =m; j*=2) {   // 沿key较大的孩子结点向下筛选
bubuko.com,布布扣
bubuko.com,布布扣    if (j < m && H.r[j].key < H.r[j+1].key) ++j; // j为key较大的记录的下标
bubuko.com,布布扣
bubuko.com,布布扣    if (rc.key >= H.r[j].key) break;         // rc应插入在位置s上
bubuko.com,布布扣
bubuko.com,布布扣    H.r[s] = H.r[j];  s = j;
bubuko.com,布布扣
bubuko.com,布布扣  }
bubuko.com,布布扣
bubuko.com,布布扣  H.r[s] = rc;  // 插入
bubuko.com,布布扣
bubuko.com,布布扣} // HeapAdjust    
bubuko.com,布布扣
bubuko.com,布布扣  
bubuko.com,布布扣
bubuko.com,布布扣void HeapSort(HeapType &H) {
bubuko.com,布布扣
bubuko.com,布布扣   // 对顺序表H进行堆排序。
bubuko.com,布布扣
bubuko.com,布布扣   int i;
bubuko.com,布布扣
bubuko.com,布布扣   RedType temp;
bubuko.com,布布扣
bubuko.com,布布扣   for (i=H.length/2; i>0; --i)  // 把H.r[1..H.length]建成大顶堆
bubuko.com,布布扣
bubuko.com,布布扣      HeapAdjust ( H, i, H.length );
bubuko.com,布布扣
bubuko.com,布布扣      for (i=H.length; i>1; --i) {
bubuko.com,布布扣
bubuko.com,布布扣         temp=H.r[i];
bubuko.com,布布扣
bubuko.com,布布扣         H.r[i]=H.r[1];
bubuko.com,布布扣
bubuko.com,布布扣         H.r[1]=temp;  // 将堆顶记录和当前未经排序子序列Hr[1..i]中
bubuko.com,布布扣
bubuko.com,布布扣                       // 最后一个记录相互交换
bubuko.com,布布扣
bubuko.com,布布扣         HeapAdjust(H, 1, i-1);  // 将H.r[1..i-1] 重新调整为大顶堆
bubuko.com,布布扣
bubuko.com,布布扣      }
bubuko.com,布布扣
bubuko.com,布布扣} // HeapSort    

 

归并排序


时间复杂度:平均情况—O(nlog2n) 最坏情况—O(nlog2n) 辅助空间:O(n) 稳定性:稳定

归并排序是把序列递归地分成短序列,递归出口是短序列只有1个元素(认为直接有序)或者2个序列(1次比较和交换),然后把各个有序的段序列合并成一个有序的长序列,不断合并直到原序列全部排好序。可以发现,在1个或2个元素时,1个元素不会交换,2个元素如果大小相等也没有人故意交换,这不会破坏稳定性。那么,在短的有序序列合并的过程中,稳定是是否受到破坏?没有,合并过程中我们可以保证如果两个当前元素相等时,我们把处在前面的序列的元素保存在结果序列的前面,这样就保证了稳定性。所以,归并排序也是稳定的排序算法。

 
C# 代码   复制

bubuko.com,布布扣void Merge (RedType SR[], RedType TR[], int i, int m, int n) {
bubuko.com,布布扣
bubuko.com,布布扣   // 将有序的SR[i..m]和SR[m+1..n]归并为有序的TR[i..n]
bubuko.com,布布扣
bubuko.com,布布扣   int j,k;
bubuko.com,布布扣
bubuko.com,布布扣   for (j=m+1, k=i;  i < =m && j < =n;  ++k) {
bubuko.com,布布扣
bubuko.com,布布扣      // 将SR中记录由小到大地并入TR
bubuko.com,布布扣
bubuko.com,布布扣      if LQ(SR[i].key,SR[j].key) TR[k] = SR[i++];
bubuko.com,布布扣
bubuko.com,布布扣      else TR[k] = SR[j++];
bubuko.com,布布扣
bubuko.com,布布扣   }
bubuko.com,布布扣
bubuko.com,布布扣   if (i < =m)  // TR[k..n] = SR[i..m];  将剩余的SR[i..m]复制到TR
bubuko.com,布布扣
bubuko.com,布布扣      while (k < =n && i < =m) TR[k++]=SR[i++];
bubuko.com,布布扣
bubuko.com,布布扣   if (j < =n)  // 将剩余的SR[j..n]复制到TR
bubuko.com,布布扣
bubuko.com,布布扣      while (k < =n &&j  < =n) TR[k++]=SR[j++];
bubuko.com,布布扣
bubuko.com,布布扣} // Merge    
bubuko.com,布布扣
bubuko.com,布布扣  
bubuko.com,布布扣
bubuko.com,布布扣void MSort(RedType SR[], RedType TR1[], int s, int t) {
bubuko.com,布布扣
bubuko.com,布布扣   // 将SR[s..t]归并排序为TR1[s..t]。
bubuko.com,布布扣
bubuko.com,布布扣   int m;
bubuko.com,布布扣
bubuko.com,布布扣   RedType TR2[20];
bubuko.com,布布扣
bubuko.com,布布扣   if (s==t) TR1[t] = SR[s];
bubuko.com,布布扣
bubuko.com,布布扣   else {
bubuko.com,布布扣
bubuko.com,布布扣      m=(s+t)/2;            // 将SR[s..t]平分为SR[s..m]和SR[m+1..t]
bubuko.com,布布扣
bubuko.com,布布扣      MSort(SR,TR2,s,m);    // 递归地将SR[s..m]归并为有序的TR2[s..m]
bubuko.com,布布扣
bubuko.com,布布扣      MSort(SR,TR2,m+1,t);  // 将SR[m+1..t]归并为有序的TR2[m+1..t]
bubuko.com,布布扣
bubuko.com,布布扣      Merge(TR2,TR1,s,m,t); // 将TR2[s..m]和TR2[m+1..t]归并到TR1[s..t]
bubuko.com,布布扣
bubuko.com,布布扣   }
bubuko.com,布布扣
bubuko.com,布布扣} // MSort    
bubuko.com,布布扣
bubuko.com,布布扣  
bubuko.com,布布扣
bubuko.com,布布扣void MergeSort(SqList &L) {
bubuko.com,布布扣
bubuko.com,布布扣  // 对顺序表L作归并排序。
bubuko.com,布布扣
bubuko.com,布布扣  MSort(L.r, L.r, 1, L.length);
bubuko.com,布布扣
bubuko.com,布布扣} // MergeSort  

 

C#排序算法的比较

标签:style   http   io   color   ar   for   sp   strong   div   

原文地址:http://www.cnblogs.com/rr163/p/4078613.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!