欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深入解析Radix Sort基數(shù)排序算法思想及C語言實現(xiàn)示例

 更新時間:2016年07月06日 16:19:41   作者:zhoutk  
基數(shù)排序和桶排序、計數(shù)排序共同是三種最常用的線性排序算法,這里我們就來深入解析Radix Sort基數(shù)排序算法思想及C語言實現(xiàn)示例,需要的朋友可以參考下

基本思想:

將待排數(shù)據(jù)中的每組關(guān)鍵字依次進(jìn)行桶分配。
具體示例:

278、109、063、930、589、184、505、269、008、083

我們將每個數(shù)值的個位,十位,百位分成三個關(guān)鍵字: 278 -> k1(個位)=8,k2(十位)=7,k3=(百位)=2。

然后從最低位個位開始(從最次關(guān)鍵字開始),對所有數(shù)據(jù)的k1關(guān)鍵字進(jìn)行桶分配(因為,每個數(shù)字都是 0-9的,因此桶大小為10),再依次輸出桶中的數(shù)據(jù)得到下面的序列。

930、063、083、184、505、278、008、109、589、269

再對上面的序列接著進(jìn)行針對k2的桶分配,輸出序列為:

505、008、109、930、063、269、278、083、184、589

最后針對k3的桶分配,輸出序列為:

008、063、083、109、184、269、278、505、589、930

效率分析:

基數(shù)排序的性能比桶排序要略差。每一次關(guān)鍵字的桶分配都需要O(N)的時間復(fù)雜度,而且分配之后得到新的關(guān)鍵字序列又需要O(N)的時間復(fù)雜度。假如待排數(shù)據(jù)可以分為d個關(guān)鍵字,則基數(shù)排序的時間復(fù)雜度將是O(d*2N) ,當(dāng)然d要遠(yuǎn)遠(yuǎn)小于N,因此基本上還是線性級別的?;鶖?shù)排序的空間復(fù)雜度為O(N+M),其中M為桶的數(shù)量。一般來說N>>M,因此額外空間需要大概N個左右。

但是,對比桶排序,基數(shù)排序每次需要的桶的數(shù)量并不多。而且基數(shù)排序幾乎不需要任何“比較”操作,而桶排序在桶相對較少的情況下,桶內(nèi)多個數(shù)據(jù)必須進(jìn)行基于比較操作的排序。因此,在實際應(yīng)用中,基數(shù)排序的應(yīng)用范圍更加廣泛。


舉例:
假設(shè)我們有一些二元組(a,b),要對它們進(jìn)行以a為首要關(guān)鍵字,b的次要關(guān)鍵字的排序。我們可以先把它們先按照首要關(guān)鍵字排序,分成首要關(guān)鍵字相同的若干堆。然后,在按照次要關(guān)鍵值分別對每一堆進(jìn)行單獨排序。最后再把這些堆串連到一起,使首要關(guān)鍵字較小的一堆排在上面。按這種方式的基數(shù)排序稱為MSD(Most Significant Dight)排序。

第二種方式是從最低有效關(guān)鍵字開始排序,稱為LSD(Least Significant Dight)排序。首先對所有的數(shù)據(jù)按照次要關(guān)鍵字排序,然后對所有的數(shù)據(jù)按照首要關(guān)鍵字排序。要注意的是,使用的排序算法必須是穩(wěn)定的,否則就會取消前一次排序的結(jié)果。由于不需要分堆對每堆單獨排序,LSD方法往往比MSD簡單而開銷小。下文介紹的方法全部是基于LSD的。

通常,基數(shù)排序要用到計數(shù)排序或者桶排序。使用計數(shù)排序時,需要的是Order數(shù)組。使用桶排序時,可以用鏈表的方法直接求出排序后的順序。下面是一段用桶排序?qū)ΧM基數(shù)排序的程序:


#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
using namespace std;
struct data
{
  int key[2];
};
struct linklist
{
  linklist *next;
  data value;
  linklist(data v,linklist *n):value(v),next(n){}
  ~linklist() {if (next) delete next;}
};
void BucketSort(data *A,int N,int K,int y)
{
  linklist *Bucket[101],*p;//建立桶
  int i,j,k,M;
  M=K/100+1;
  memset(Bucket,0,sizeof(Bucket));
  for (i=1;i<=N;i++)
  {
    k=A[i].key[y]/M; //把A中的每個元素按照的范圍值放入對應(yīng)桶中
    Bucket[k]=new linklist(A[i],Bucket[k]);
  }
  for (k=j=0;k<=100;k++)
  {
    for (p=Bucket[k];p;p=p->next) j++;
    for (p=Bucket[k],i=1;p;p=p->next,i++)
      A[j-i+1]=p->value; //把桶中每個元素取出
    delete Bucket[k];
  }
}
void RadixSort(data *A,int N,int K)
{
  for (int j=1;j>=0;j--) //從低優(yōu)先到高優(yōu)先 LSD
    BucketSort(A,N,K,j);
}
int main()
{
  int N=100,K=1000,i;
  data *A=new data[N+1];
  for (i=1;i<=N;i++)
  {
    A[i].key[0]=rand()%K+1;
    A[i].key[1]=rand()%K+1;
  }
  RadixSort(A,N,K);
  for (i=1;i<=N;i++)
    printf("(%d,%d) ",A[i].key[0],A[i].key[1]);
  printf("\n");
  return 0;
}

基數(shù)排序是一種用在老式穿卡機(jī)上的算法。一張卡片有80列,每列可在12個位置中的任一處穿孔。排序器可被機(jī)械地"程序化"以檢查每一迭卡片中的某一列,再根據(jù)穿孔的位置將它們分放12個盒子里。這樣,操作員就可逐個地把它們收集起來。其中第一個位置穿孔的放在最上面,第二個位置穿孔的其次,等等。

對于一個位數(shù)有限的十進(jìn)制數(shù),我們可以把它看作一個多元組,從高位到低位關(guān)鍵字重要程度依次遞減??梢允褂没鶖?shù)排序?qū)σ恍┪粩?shù)有限的十進(jìn)制數(shù)排序。

相關(guān)文章

  • 使用C語言打印月歷

    使用C語言打印月歷

    這篇文章主要為大家詳細(xì)介紹了使用C語言打印月歷,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-06-06
  • C++ Boost EnableIf函數(shù)使用介紹

    C++ Boost EnableIf函數(shù)使用介紹

    Boost是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱。Boost庫是一個可移植、提供源代碼的C++庫,作為標(biāo)準(zhǔn)庫的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開發(fā)引擎之一,是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱
    2022-11-11
  • Qt控件點擊消息獲取的方法詳解

    Qt控件點擊消息獲取的方法詳解

    本文將利用Qt中的QLabel、QPushButton這兩個控件,為大家詳細(xì)介紹一下Qt控件點擊消息獲取的方法,文中的示例代碼講解詳細(xì),感興趣的可以了解一下
    2022-06-06
  • C語言實現(xiàn)校運動會項目管理系統(tǒng)

    C語言實現(xiàn)校運動會項目管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C語言實現(xiàn)校運動會項目管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • C語言中settimeofday函數(shù)和gettimeofday函數(shù)的使用

    C語言中settimeofday函數(shù)和gettimeofday函數(shù)的使用

    這篇文章主要介紹了C語言中的settimeofday函數(shù)和gettimeofday函數(shù)的使用,注意settimeofday()函數(shù)只返回0和-1,需要的朋友可以參考下
    2015-08-08
  • Opencv光流運動物體追蹤詳解

    Opencv光流運動物體追蹤詳解

    這篇文章主要為大家詳細(xì)介紹了Opencv光流運動物體追蹤的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • C++非遞歸遍歷磁盤文件和遞歸遍歷磁盤文件的程序示例

    C++非遞歸遍歷磁盤文件和遞歸遍歷磁盤文件的程序示例

    這篇文章主要介紹了C++非遞歸遍歷磁盤文件和遞歸遍歷磁盤文件的程序示例,大家可以參考使用二種方法
    2013-11-11
  • C語言實現(xiàn)字母大小寫轉(zhuǎn)換的方法

    C語言實現(xiàn)字母大小寫轉(zhuǎn)換的方法

    這篇文章主要介紹了C語言實現(xiàn)字母大小寫轉(zhuǎn)換的方法,涉及C語言字符串的遍歷與轉(zhuǎn)換技巧,非常簡單實用,需要的朋友可以參考下
    2015-07-07
  • C++ 隨機(jī)數(shù)與隨機(jī)種子數(shù)的實例

    C++ 隨機(jī)數(shù)與隨機(jī)種子數(shù)的實例

    這篇文章主要介紹了C++ 隨機(jī)數(shù)與隨機(jī)種子數(shù)的實例的相關(guān)資料,需要的朋友可以參考下
    2017-07-07
  • C++如何調(diào)用已經(jīng)寫好的C接口

    C++如何調(diào)用已經(jīng)寫好的C接口

    如何在C++代碼中調(diào)用寫好的C接口?你可能會奇怪,C++不是兼容C嗎?直接調(diào)用不就可以了,那么我們來測試一下,先看看C++如何調(diào)用C代碼接口的,需要的朋友可以參考一下
    2021-10-10

最新評論