幾種排序算法
阿新 • • 發佈:2018-12-30
ace 基準 n) swap d+ urn 解決 隨機選擇 ges
快速排序
快速排序是一種基於分治的算法,其基本思想是將一個大數組按照一個【基準數】分為左右兩份,左邊的部分都不大於基準數,右邊的部分都不小於基準數。然後,對這兩份在分別應用快速排序,直到剩下一個元素為止。快速排序的平均時間復雜度為nlog(n)。
下面是選取數組最左邊的元素為基準元素的快排算法:
//應該隨機選擇一個數為主元,這裏選擇最右邊的一個數為主元 ## 快速排序 int partition(int a[],int left,int right){ int temp=a[left];//選擇數組最左邊的元素為基準 while(left<right){ while(left<right&&a[right]>temp) right--; a[left]=a[right]; while(left<right&&a[left]<=temp) left++; a[right]=a[left]; } a[left]=temp; return left; } //法二 int partition(int a[],int left,int right){ int small=left-1;//記錄比主元小的數 for(int i=left;i<right;i++){ if(a[i]<a[right]){ ++small; if(small!=i){ swap(a[i],a[small]); } } } small++; swap(a[small],a[right]); return small; } void quickSort(int a[],int left,int right){ if(left<right){ int pos=partion(a,left,right); quickSort(a,left,pos-1); quickSort(a,pos+1,right); } }
快速排序算法當序列中元素的排列比較隨機時效率最高,但是當序列中元素接近有序時,會達到最壞的時間復雜度O(n2),產生這種情況的主要原因在於沒有把當前區間劃分為兩個長度接近的子區間。
解決辦法就是隨機選取基準元素,也就是對A[left,...,right]來說,不總是用A[left]作為基準元素,而是從A[left],A[left+1].....,A[right]中隨機選擇一個作為基準元素。
這樣對於任意輸入數據的期望時間復雜度就能達到O(nlogn)。
另外快速排序是不穩定的。
堆排序
堆排序是使用堆這種數據結構來實現的,首先用一個數組實現堆,主要的步驟是向下調整的過程,如下:
#include<cstdio> #include<algorithm> using namespace std; const int maxn=10010; int heap[maxn]={-1,9,8,1,2,6,7,4,3,5,0},n=10; void downAdjust(int low,int high){ int i=low,j=i*2; while(j<=high){ if(j+1<=high&&heap[j+1]>heap[j]){ j++; } if(heap[j]>heap[i]){ swap(heap[j],heap[i]); i=j; j=i*2; }else{ break; } } } void createHeap(){ for(int i=n/2;i>=1;i--){ downAdjust(i,n); } } void heapSort(){ createHeap(); for(int i=n;i>1;i--){ swap(heap[i],heap[1]); downAdjust(1,i-1); } } int main(){ heapSort(); for(int i=1;i<=n;i++){ printf("%d ",heap[i]); } return 0; }
歸並排序
void merge(int a[],int L1,int R1,int L2,int R2){ int i=L1,j=L2; int temp[maxn],index=0; while(i<=R1&&j<=R2){ if(a[i]<a[j]){ temp[index++]=a[i++]; }else{ temp[index++]=a[j++]; } } while(i<=R1) temp[index++]=a[i++]; while(j<=R2) temp[index++]=a[j++]; for(i=0;i<index;i++){ a[L1+1]=temp[i]; } } // 遞歸實現 void mergeSort(int a[],int left,int right){ if(left<right){ int mid=(left+right)/2; mergeSort(a,left,mid); mergeSort(a,mid+1,rigt); mergeSort(a,left,mid,mid+1,right); } } // 非遞歸實現 void mergeSort(int a[]){ for(int step=2;step/2<n;stem*=2){ for(int i0;i<n;i+=step){ int mid=i+step/2-1 if(mid+1<n){ merge(a,i,mid,mid+1,min(i+stemp-1,n-1)); } } } }
幾種排序算法