Algorithm --> 十大排序算法

十大排序算法

 

主要排序法有:  
一、冒泡( Bubble)排序—— 相邻交换  
二、选择排序 ——每次最小/ 大排在相应的位置  
三、插入排序 ——将下一个插入已排好的序列中  
四、壳( Shell)排序—— 缩小增量  
五、归并排序  
六、快速排序  (待补充)
七、堆排序  (待补充)
八、拓扑排序  (待补充)
九、锦标赛排序  (待补充)
十、基数排序  (待补充)


 

 

1、冒泡( Bubble)排序 

基本思想:在要排序的一组数中,对当前还未排好序的范围内的全部数,自上而下对相邻的两个数依次进行比较和调整,让较大的数往下沉,较小的往上冒。即:每当两相邻的数比较后发现它们的排序与排序要求相反时,就将它们互换。

void BubbleSortArray() 
{ 
    for(int i=1;i<n;i++) 
    { 
        for(int j=0;i<n-i;j++) 
        { 
            if(a[j]>a[j+1])//比较交换相邻元素 
            { 
                int temp; 
                temp=a[j];
                a[j]=a[j+1];
                a[j+1]=temp; 
            } 
        } 
    } 
} 

效率 O , 适用于排序小列表。

 

2、选择排序

基本思想:在要排序的一组数中,选出最小的一个数与第一个位置的数交换;然后在剩下的数当中再找最小的与第二个位置的数交换,如此循环到倒数第二个数和最后一个数比较为止。

void SelectSortArray() 
{ 
    int min_index; 
    for(int i=0;i<n-1;i++) 
    { 
        min_index=i; 
        for(int j=i+1;j<n;j++)//每次扫描选择最小项 
        {
              if(arr[j]<arr[min_index])  
                min_index=j;
        } 
        if(min_index!=i)//找到最小项交换,即将这一项移到列表中的正确位置  
        { 
            int temp; 
            temp=arr[i];
            arr[i]=arr[min_index];
            arr[min_index]=temp; 
        } 
    } 
} 

效率O),适用于排序小的列表。

 

 

3、 插入排序

基本思想:在要排序的一组数中,假设前面(n-1) [n>=2] 个数已经是排好顺序的,现在要把第n个数插到前面的有序数中,使得这n个数也是排好顺序的。如此反复循环,直到全部排好顺序。

void InsertSortArray() 
{ 
    for(int i=1;i<n;i++)//循环从第二个数组元素开始,因为 arr[0]作为最初已排序部分 
    { 
        int temp=arr[i];//temp标记为未排序第一个元素  
        int j=i-1; 
        while (j>=0 && arr[j]>temp)/*将 temp与已排序元素从小到大比较,寻找 temp应插入的位置*/ 
        { 
            arr[j+1]=arr[j]; 
            j--; 
        } 
        arr[j+1]=temp; 
    } 
} 

最佳效率 On );最糟效率 O )与冒泡、选择相同,适用于排序小列表若列表基本有序,则插入排序比冒泡、选择更有效率。

 

4、希尔(Shell)排序

基本思想:算法先将要排序的一组数按某个增量dn/2,n为要排序数的个数)分成若干组,每组中记录的下标相差d.对每组中全部元素进行直接插入排序,然后再用一个较小的增量(d/2)对它进行分组,在每组中再进行直接插入排序。当增量减到1时,进行直接插入排序后,排序完成。

void  ShellSortArray1() 
{ 
    for(int incr=3;incr>0;incr--)//增量递减,以增量 3,2 ,1为例  
    { 
        for(int L=0;L<(n-1)/incr;L++)//重复分成的每个子列表 
        { 
            for(int i=L+incr;i<n;i+=incr)//对每个子列表应用插入排序  
            { 
                int temp=arr[i]; 
                int j=i-incr; 
                while(j>=0&&arr[j]>temp) 
                { 
                    arr[j+incr]=arr[j]; 
                    j-=incr; 
                } 
                arr[j+incr]=temp; 
            } 
        } 
    } 
} 

void  ShellSortArray2(int *data, int len) 
{ 
    int d = len;
    while(d > 1)
    {
        d = (d+1)/2;
        for(int i=0; i<len-d; i++)
        {
            if(data[i+d] < data[i])
            {
                int tmp = data[i+d];
                data[i+d] = data[i];
                data[i] = tmp;
            }
        }
    }
} 

适用于排序小列表。 效率估计 Onlog2^n ~O n^1.5),取决于增量值的最初大小。建议使用质数作为增量值,因为如果增量值是 2的幂,则在下一个通道中会再次比较相同的元素。 壳(Shell)排序改进了插入排序,减少了比较的次数。是不稳定的排序,因为排序过程中元素可能会前后跳跃。  

 

5、归并排序

基本思想:归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的。然后再把有序子序列合并为整体有序序列。

//将有二个有序数列a[first...mid]和a[mid...last]合并。  
void mergearray(int a[], int first, int mid, int last, int temp[])  
{  
    int i = first, j = mid + 1;  
    int m = mid,   n = last;  
    int k = 0;  
      
    while (i <= m && j <= n)  
    {  
        if (a[i] <= a[j])  
            temp[k++] = a[i++];  
        else  
            temp[k++] = a[j++];  
    }  
    while (i <= m)  temp[k++] = a[i++];  
    while (j <= n)  temp[k++] = a[j++];  
    for (i = 0; i < k; i++)  
        a[first + i] = temp[i];  
} 

void mergesort(int a[], int first, int last, int temp[])  
{  
    if (first < last)  
    {  
        int mid = (first + last) / 2;  
        mergesort(a, first, mid, temp);    //左边有序  
        mergesort(a, mid + 1, last, temp); //右边有序  
        mergearray(a, first, mid, last, temp); //再将二个有序数列合并  
    }  
}  
  
bool MergeSort(int a[], int n)  
{  
    int *p = new int[n];  
    if (p == NULL)  return false;  
    mergesort(a, 0, n - 1, p);  
    delete[] p;  
    return true;  
} 

 

posted @ 2015-10-25 19:47  蚂蚁吃大象、  阅读(437)  评论(0编辑  收藏  举报