希尔排序

希尔排序的实质就是分组插入排序,该方法又称缩小增量排序,因DL.Shell于1959年提出而得名。

 

  插入排序的算法复杂度为O(n2),但如果序列为正序可提高到O(n),而且直接插入排序算法比较简单,希尔排序利用这两点得到了一种改进后的插入排序。

一. 算法描述

希尔排序:将无序数组分割为若干个子序列,子序列不是逐段分割的,而是相隔特定的增量的子序列,对各个子序列进行插入排序;然后再选择一个更小的增量,再将数组分割为多个子序列进行排序......最后选择增量为1,即使用直接插入排序,使最终数组成为有序。
增量的选择:在每趟的排序过程都有一个增量,至少满足一个规则 增量关系 d[1] > d[2] > d[3] >..> d[t] = 1 (t趟排序);根据增量序列的选取其时间复杂度也会有变化,这个不少论文进行了研究,在此处就不再深究; 本文采用首选增量为n/2,以此递推,每次增量为原先的1/2,直到增量为1;
下图详细讲解了一次希尔排序的过程:



二. 算法分析

平均时间复杂度:希尔排序的时间复杂度和其增量序列有关系,这涉及到数学上尚未解决的难题;不过在某些序列中复杂度可以为O(n1.3);

空间复杂度:O(1)  

稳定性:不稳定

三. 算法实现

[cpp]  view plain copy
  1. /******************************************************** 
  2. *函数名称:ShellInsert 
  3. *参数说明:pDataArray 无序数组; 
  4. *          d          增量大小 
  5. *          iDataNum为无序数据个数 
  6. *说明:    希尔按增量d的插入排序 
  7. *********************************************************/  
  8. void ShellInsert(int* pDataArray, int d, int iDataNum)  
  9. {  
  10.     for (int i = d; i < iDataNum; i += 1)    //从第2个数据开始插入  
  11.     {  
  12.         int j = i - d;  
  13.         int temp = pDataArray[i];    //记录要插入的数据  
  14.         while (j >= 0 && pDataArray[j] > temp)    //从后向前,找到比其小的数的位置  
  15.         {  
  16.             pDataArray[j+d] = pDataArray[j];    //向后挪动  
  17.             j -= d;  
  18.         }  
  19.   
  20.         if (j != i - d)    //存在比其小的数  
  21.             pDataArray[j+d] = temp;  
  22.     }  
  23. }  
  24.   
  25. /******************************************************** 
  26. *函数名称:ShellSort 
  27. *参数说明:pDataArray 无序数组; 
  28. *          iDataNum为无序数据个数 
  29. *说明:    希尔排序 
  30. *********************************************************/  
  31. void ShellSort(int* pDataArray, int iDataNum)  
  32. {  
  33.     int d = iDataNum / 2;    //初始增量设为数组长度的一半  
  34.     while(d >= 1)  
  35.     {  
  36.         ShellInsert(pDataArray, d, iDataNum);  
  37.         d = d / 2;    //每次增量变为上次的二分之一  
  38.     }  
  39. }  

  

附注:上面希尔排序的步长选择都是从n/2开始,每次再减半,直到最后为1。其实也可以有另外的更高效的步长选择

附上自己的程序:

#include <stdio.h>
#include <iostream>
using namespace std;
//其实质就是通过步长的方式逐次处理数据,然后每个序列内部的数据采用插入排序的方式
/******************************************************** 
*函数名称:ShellInsert 
*参数说明:pDataArray 无序数组; 
*          d          增量大小 
*          iDataNum为无序数据个数 
*说明:    希尔按增量d的插入排序 
*********************************************************/  
void ShellInsert(int* pDataArray, int d, int iDataNum)  
{  
    for (int i = d; i < iDataNum; i += 1)    //从第2个数据开始插入  
    {  
        int j = i - d;  
        int temp = pDataArray[i];    //记录要插入的数据  
        while (j >= 0 && pDataArray[j] > temp)    //从后向前,找到比其小的数的位置  
        {  
            pDataArray[j+d] = pDataArray[j];    //向后挪动  
            j -= d;  
        }  
  
        if (j != i - d)    //存在比其小的数  
            pDataArray[j+d] = temp;  
    }  
}  
/******************************************************** 
*函数名称:ShellSort 
*参数说明:pDataArray 无序数组; 
*          iDataNum为无序数据个数 
*说明:    希尔排序 
*********************************************************/  
void ShellSort(int* pDataArray, int iDataNum)  
{  
    int d = iDataNum / 2;    //初始增量设为数组长度的一半  
    while(d >= 1)  
    {  
        ShellInsert(pDataArray, d, iDataNum);  
        d = d / 2;    //每次增量变为上次的二分之一  
    }  
}  
void main()
{
int arrays[]={3423,43,5,54,65,2,3,65,43};
ShellSort(arrays,sizeof(arrays)/sizeof(int));
for(int i=0;i<sizeof(arrays)/sizeof(int);i++)
{
cout<<arrays[i]<<" ";
}
system("pause");
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值