zoukankan      html  css  js  c++  java
  • 算法-一步步教你如何用c语言实现堆排序(非递归)

    看了左神的堆排序,觉得思路很清晰,比常见的递归的堆排序要更容易理解,所以自己整理了一下笔记,带大家一步步实现堆排序算法

    首先介绍什么是大根堆:每一个子树的最大值都是子树的头结点,即根结点是所有结点的最大值

    堆排序是基于数组和二叉树思想实现的(二叉树是脑补结构,实际是数组)

    堆排序过程

    1、数组建成大根堆,首先,遍历所有结点,当前结点index和父结点(index-1)/ 2 比较 (当前数组的下标是index,此结点的父结点的下标就是(index-1)/ 2  ),如果比父结点大,交换,变成父结点的位置再和上一层的父结点比较,直到满足大根堆条件

    int swap(int source[], int a, int b)
    {
        int temp = source[a];
        source[a] = source[b];
        source[b] = temp;
    }
    int heapsort(int source[], int len)
    {
        for (int i = 0; i <len; i++)
        {
            heapInsert(source, i);
        }
    }
    int heapInsert(int source[], int index)
    {
        while (source[index] > source[(index - 1) / 2])
        {
            swap(source, index, (index - 1) / 2);
            index = (index - 1) / 2;
        }
    }

    2、让根结点和最后一个结点交换位置,也就是数组的第一个数和最后一个数交换位置,接下来最后一个数不用考虑了,比如一个数组有5个数,定义一个变量size=5,大根堆的根结点放到最后一个数后,--size

    int size = len;
    swap(source, 0, --size);

    3、让交换后的头结点经历一个向下的调整,让结点和自己的两个孩子比较,如果孩子的值比头结点大,交换,交换到孩子结点位置,继续和下面的两个孩子比较,直到满足大根堆条件

    int heapify(int source[], int index, int size)//index表示要和它两个孩子比较的结点
    {
        int left = index * 2 + 1;  //找到index左孩子结点的数组下标
        while (left < size)
        {
            int largest = left + 1 < size && source[left + 1] > source[left] ? source[left + 1] : source[left];//如果有右孩子且右孩子比左孩子大,令largest=右孩子的值,也就是把两个孩子中最大的一个数赋给largest
            if (source[index] < source[largest])
            {
                swap(source, index, largest);
                index = largest;
                left = index * 2 + 1;
            }
            else break;
        }
    
    }

    4、重复第2步,第3步,直到size = 0 ,整个数组排序过程结束

    while (size > 0)
        {
            swap(source, 0, --size);
            heapify(source, 0, size);
        }

    源代码如下

    #define _CRT_SECURE_NO_WARNINGS
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    int swap(int source[], int a, int b)
    {
        int temp = source[a];
        source[a] = source[b];
        source[b] = temp;
    }
    int heapsort(int source[], int len)
    {
        for (int i = 0; i < len; i++)
        {
            heapInsert(source, i);
        }
        int size = len;
        while (size > 0)
        {
            swap(source, 0, --size);
            heapify(source, 0, size);
        }
    }
    int heapInsert(int source[], int index)
    {
        while (source[index] > source[(index - 1) / 2])
        {
            swap(source, index, (index - 1) / 2);
            index = (index - 1) / 2;
        }
    }
    int heapify(int source[], int index, int size)//index表示要和它两个孩子比较的结点
    {
        int left = index * 2 + 1;  //找到index左孩子结点的数组下标
        while (left < size)
        {
            int largest = left + 1 < size && source[left + 1] > source[left] ? left + 1 : left;//如果有右孩子且右孩子比左孩子大,令largest=右孩子的值,也就是把两个孩子中最大的一个数赋给largest
            if (source[index] < source[largest])
            {
                swap(source, index, largest);
                index = largest;
                left = index * 2 + 1;
            }
            else break;
        }
    
    }
    int main()
    {
        int source[] = { 10,16,123,8,79,6,54,65,48,6,54,536,654,64,8,9,25,17 };
        int len;
        len = sizeof(source) / sizeof(int);
        heapsort(source, len);
        for (int i = 0; i < len; i++)
        {
            printf("%d ", source[i]);
        }
        
    }

    输出结果

    以上就是堆排序的所有细节,这个版本很优良,堆排序的额外空间复杂度是O(1),如果用递归的话,递归有递归栈,额外空间复杂度不就上去了吗,设计成这种迭代的可以省空间,时间复杂度为O(n log n)

    转载请注明出处、作者  谢谢

  • 相关阅读:
    angularjs 学习小结
    .NET 环境中使用RabbitMQ
    .NET 使用 Azure Blob 存储图片或文件
    PL/SQL Developer如何连接64位的Oracle图解
    安装Visual Studio 语言包时出现windows 程序兼容模式已打开.请将其关闭
    .NET Core使用Swagger视图时,出现undefined/swagger/v2/swagger.json的错误
    当gitlab密码修改后,无法拉取代码,提交推送代码。如何在本地修改正确密码
    C#对配置文件的,增,删,改,查
    C#log4net系统日志
    C#工厂模式-工厂方法
  • 原文地址:https://www.cnblogs.com/cyyz-le/p/11252193.html
Copyright © 2011-2022 走看看