希尔排序与堆排序

希尔排序

介绍

希尔排序由希尔发明,当一个序列的无序程度较低时,那么使用插入排序的效率就会很快,希尔排序的核心思想就是,通过逐步缩小增量(步长)来逐步把一开始很无序的序列慢慢变得有序,最后当增量为1时,就是一次典型的插入排序。

实现

cpp 复制代码
typedef int ElementType;

void ShellSort(ElementType A[], int N) {
    int i,j,Increment;
    ElementType tmp;
    for (Increment=N/2; Increment>0; Increment/=2) {
        for (i=Increment; i<N; i++) {
            tmp = A[i];
            for (j=i; j>=Increment; j-=Increment) {
                if (tmp < A[j-Increment]) {
                    A[j] = A[j-Increment];
                }else {
                    break;
                }
            }
            A[j] = tmp;
        }
    }
}

假如有一个元素数量为11的乱序序列,那么就可以得到11/2=5,5/2=2,2/1=1,这三个增量。对序列进行增量为5的插入排序,分别获得下标为(5,0),(6,1),(7,2),(8,3),(9,4),(10,5)这六组下标的跳跃步长为5的序列,然后分别对这六组序列进行插入排序,排完序后的整个数列的有序程度就会升高一些,这样就可以提高之后进行直接插入排序的效率了。

接着进行增量为2的插入排序,分别获得下标为(2,0),(3,1),(4,2,0),(5,3,1),(6,4,2,0),(7,5,3,1),(8,6,4,2,0),(9,7,5,3,1),(10,8,6,4,2,0)这九组下标的跳跃步长为2的序列,再分别对这九组序列进行插入排序,排序完后整个序列的有序程度就很高了。看似这一轮的比较次数很多,但其实还有个break语句,其表示若第一次比较不交换时,那就不进行后续的比较了,例如(4,2,0),若下标为4和2的不需要交换,那说明下标为4,2,0的就已经是有序了,不需要再对2,0进行比较了。若下标为4,2的进行交换了,那还得继续比较下标为4和0的元素。

算法中第一个for循环得到增量矩阵,第二个for循环获得各个分序列中的首元素的下标,第三个for循环对各个分序列进行插入排序。

分析

希尔排序是不稳定排序,其会把不同步长的分序列进行排序,当相同元素被分到不同分序列中时,可能会改变它们的相对位置。

不同步长序列的时间复杂度不一样,若为n/2,n/4....1的步长的话,那最坏的时间复杂度为O().若用Hibbard步长序列(-1),那时间复杂度为O().若用Sedgewick步长序列(混合序列9×-9×+1和-3×+1),那最坏的情况下时间复杂度被优化成O()了。

堆排序

介绍

在之前的文章中介绍了堆这一数据结构,得益于堆的特性,其根节点始终是最大值(或最小值),我们可以不停的取堆的根节点元素然后按顺序排好,这样就可以得到排好序的序列了。

实现

cpp 复制代码
typedef int ElementType;
#define LeftChild(i) (2*(i)+1)

void Swap(ElementType *a, ElementType *b) {
    ElementType temp = *a;
    *a = *b;
    *b = temp;
}

void PercDown(ElementType A[],int i,int N) {
    int Child;
    ElementType Tmp;
    for (Tmp = A[i];LeftChild(i)<N;i=Child) {
        Child = LeftChild(i);
        if (Child!=N-1 && A[Child+1]>A[Child]) {
            Child++;
        }
        if (Tmp<A[Child]) {
            A[i] = A[Child];
        }else {
            break;
        }
    }
    A[i] = Tmp;
}

void HeapSort(ElementType A[],int N) {
    int i;
    for (i=N/2;i>=0;i--) {
        PercDown(A,i,N);
    }
    for (i=N-1;i>0;i--) {
        Swap(&A[0],&A[i]);
        PercDown(A,0,i);
    }
}

得益于堆对应完全二叉树父子节点位置的数学关系,我们可以通过对原始无序序列进行下滤操作从而将其转化成符合堆结构的数组。

此时这个数组的第一个元素是整个序列的最大值,把它和最后的元素进行交换,这样整个序列中,我们完成了最大元素的归位。然后在把除最后元素的其他所有元素进行下滤操作重新形成新堆,再把第一个元素放到倒数第二个位置上,这样就完成了第二个最大元素的归位,循环往复就以从大到小的顺序完成了排序操作。

分析

堆排序是不稳定的,当把栈顶元素和最后有个元素进行交换时,可能会改变相同元素的相对位置。

要进行n-1次交换,每次交换都得重新通过下滤操作构建堆结构(该构建操作的时间复杂度为 O(LogN)),所以一共是O(NLogN)。

相关推荐
IAR Systems2 分钟前
在IAR Embedded Workbench for Renesas RH850中实现ROPI
linux·运维·算法
一个不知名程序员www17 分钟前
算法学习入门--- set与map(C++)
c++·算法
鸿途优学-UU教育25 分钟前
法考命题趋势解读:为何越来越重视“实战能力”?
算法·法律·uu教育·法考机构
Rui_Freely27 分钟前
Vins-Fusion之 相机—IMU在线标定(两帧间旋转估计)(十)
人工智能·算法·计算机视觉
独自破碎E28 分钟前
链表中的节点每k个一组翻转
数据结构·链表
mit6.8241 小时前
二分猜答案
算法
_OP_CHEN1 小时前
【算法基础篇】(四十二)数论之欧拉函数深度精讲:从互质到数论应用
c++·算法·蓝桥杯·数论·欧拉函数·算法竞赛·acm/icpc
Eloudy1 小时前
模板函数动态库与头文件设计示例
算法·cuda
星云数灵1 小时前
大模型高级工程师考试练习题4
人工智能·算法·机器学习·大模型·大模型考试题库·阿里云aca·阿里云acp大模型考试题库
千金裘换酒1 小时前
Leetcode 二叉树中序遍历 前序遍历 后序遍历(递归)
算法·leetcode·职场和发展