排序演算法2

2021-08-07 11:03:04 字數 4763 閱讀 9556

這是《漫談經典排序演算法系列》第二篇,解析了各種插入排序演算法

。主要包括:直接插入排序、折半插入排序、表插入排序、希爾插入排序。

每一種演算法的開頭都敘述了引出該演算法的原因,然後給出**,最後分析演算法效率及和其他插入排序相比,優劣在**。

各種排序演算法的解析請參考如下:

《漫談經典排序演算法:一、從簡單選擇排序到堆排序的深度解析》

《漫談經典排序演算法:二、各種插入排序解析及效能比較》

《漫談經典排序演算法:三、氣泡排序 && 快速排序》

《漫談經典排序演算法:四、歸併排序》

《漫談經典排序演算法:五、線性時間排序(計數、基數、桶排序)》

《漫談經典排序演算法:六、各種排序演算法總結》

注:為了敘述方便,本文以及源**中均不考慮a[0],預設下標從1開始。

給定待排序序列a[ 1.....n ],現假設a[1...i]已經有序,那麼我們取出a[i+1]插入到序列a[1...i].這樣有序序列記錄數就增加了1.如此重複上述操作,不斷取出記錄插入有序序列,直到a[n]插入到有序序列,排序完成。

[cpp]view plain

copy

//直接插入排序

void

straightinsertsort(

int*a,

intn)  

else

break

;  }  

a[j+1]=temp;  

}  }  

容易看出,要插入的記錄個數為n-1,其中關鍵字的比較次數和記錄移動次數是依賴於給出的待排序序列是否基本有序。在最佳情況下(待排

序序列有序),比較次數和移動次數時間為o(1),所以時間複雜度為o(n).在最壞情況下(待排序序列逆序)和平均時間均為o(n^2).

從上述分析中可以看出,直接插入排序適合記錄數比較少、給定序列基本有序的情況。熟悉了排序過程我們發現,直接插入排序是一種穩定的原地排序演算法。

在直接插入排序過程中,我們是把乙個記錄插入到有序序列中,至於要插入到有序序列中的哪個位置呢?採用的是順序查詢確定插入的位置。顯然對於有序序列,折半查詢的效率要高,所以在尋找插入位置時可以用折半查詢。折半查詢主要分為三步:1、查詢要插入的位置  2、移位  3、把記錄插入相應的位置。

[cpp]view plain

copy

//折半查詢

intbinarysearch(

int*a,

intlow,

inthigh,

intkey)  

//折半插入排序

void

binaryinsertsort(

int*a,

intn)  

}  

折半插入排序是對直接插入排序的一種改進,這種改進只考慮了關鍵字比較次數,並沒有減少移位次數,所以平均時間和最壞情況下(待排序序列逆序)時間複雜度o(n^2),如果記錄數量很大的話,這兩種情況下是優於直接插入排序。再來看一下最佳情況(待排序序列有序),此時關鍵字比較次數並不為o(1),時間複雜度為o(n*log2n)。(其中折半查詢時間複雜度o(log2n),這個在以後寫查詢的時候再分析,這裡不做詳細講解。)。所以在記錄數較小、待排序序列基本有序情況下直接插入排序優於折半插入排序。此外,折半插入排序是不穩定的原地排序,實現起來也較複雜。

折半插入排序相對於直接插入排序來說減少了比較次數。那麼我們可不可以減少移動次數呢,答案是可以的。於是就有了表插入排序,用乙個靜態鍊錶來儲存待排序序列,其他操作和直接插入排序很像。主要步驟:1、初始化鍊錶  2、取出要插入的記錄 3、遍歷鍊錶尋找插入位置  4、記錄插入鍊錶中。

[cpp]view plain

copy

//靜態鍊錶

typedef

struct

node,*pnode;  

//表插入排序

void

tableinsertsort(pnode list,

intn)  

if(p==0)else

}  }  

表插入排序也是對直接插入排序的一種改進,這種改進只減少了移動次數,並沒有減少關鍵字比較次數,所以平均時間和最壞情況下(待排序序列逆序)時間複雜度o(n^2),如果記錄數量很大的話,這兩種情況下是優於直接插入排序。再來看一下最佳情況(待排序序列有序),關鍵字比較次數並為o(1),時間複雜度為o(n)。此時和直接插入排序時間複雜度一樣。此外,表插入排序改變了記錄的儲存結構,無法順序訪問,是一種穩定的排序演算法,實現起來也較複雜。

上述兩種排序都是只考慮減少關鍵字比較次數或者只考慮減少關鍵字移動次數。有沒有別的改進辦法呢?我們注意到,直接插入排序適合於記錄數較少、基本有序的情況。於是我們可以先將整個待排序序列分割成若干子串行分別進行直接插入排序,整個序列基本有序時,再對序列進行一次直接插入排序。這就是希爾排序。

[cpp]view plain

copy

//一趟增量為dk的希爾插入排序

void

shellinsert(

int*a,

intn,

intdk)  

}  //希爾排序

void

shellsort(

int*a,

intn)  

;  for

(i=0;i<5;i++)  

shellinsert(a,6,dk[i]);  

}  

當給定序列記錄量較大時,希爾排序效能優於直接插入排序。再希爾排序的過程中,關鍵字是跳躍式移動的,這樣就減少了移動次數。希爾排序效能的分析是乙個複雜的問題,時間與所取的增量有關。增量選取的不好可能會大大降低排序效率。

《資料結構》嚴蔚敏版

[cpp]view plain

copy

#include

//直接插入排序

void

straightinsertsort(

int*a,

intn)  

else

break

;  }  

a[j+1]=temp;  

}  }  

void

main()  

;//不考慮a[0]

straightinsertsort(a,6);  

for(i=1;i<=6;i++)  

printf("%-4d"

,a[i]);  

printf("\n"

);  

}  

[cpp]view plain

copy

#include

//折半查詢

intbinarysearch(

int*a,

intlow,

inthigh,

intkey)  

//折半插入排序

void

binaryinsertsort(

int*a,

intn)  

}  void

main()  

;//不考慮a[0]

binaryinsertsort(a,6);  

for(i=1;i<=6;i++)  

printf("%-4d"

,a[i]);  

printf("\n"

);  

}  

[cpp]view plain

copy

#include

#define max 10000

//靜態鍊錶

typedef

struct

node,*pnode;  

//表插入排序

void

tableinsertsort(pnode list,

intn)  

if(p==0)else

}  }  void

main()  

;  tableinsertsort(list,6);  

p=list[0].next;  

while

(p!=0)  

printf("\n"

);  

}  

[cpp]view plain

copy

#include

//一趟增量為dk的希爾插入排序

void

shellinsert(

int*a,

intn,

intdk)  

}  //希爾排序

void

shellsort(

int*a,

intn)  

;  for

(i=0;i<5;i++)  

shellinsert(a,6,dk[i]);  

}  void

main()  

;//不考慮a[0]

shellsort(a,6);  

for(i=1;i<=6;i++)  

printf("%-4d"

,a[i]);  

printf("\n"

);  

}

排序 排序演算法2

思想 堆,堆頂元素 即第乙個元素 必為最小項 小頂堆 公升序序列 或者最大項 大頂堆 降序序列 若以一維陣列儲存乙個堆,則堆對應一棵完全二叉樹,且所有非葉結點的值均不大於 或不小於 其子女的值,根結點 堆頂元素 的值是最小 或最大 的。雖然說是一棵完全二叉樹,但不是說就得根據這些資料還需要重新建立資...

排序演算法 2

歸併排序 merge sort 依舊先上源 include include void merge int sourcearr,int temparr,int startindex,int midindex,int endindex while i midindex 1 temparr k sourc...

排序演算法 2

快速排序 quicksort 演示 現在左側是比70小的數,右側是比70大的數。繼續以左側陣列為例解釋,基準位歸位 while i j 1 arr begin arr j arr j tmp quick sort arr,begin,j 1 quick sort arr,j 1,end public...