堆排序算法学习

堆排序

堆性质:

1.是一个顺序存储的完全二叉树
2.每个节点都大于或等于它的任意子节点是大顶堆,反之是小顶堆。

算法思想(以大顶堆为例):

1.首先是一个无序的数组,画成完全二叉树的形式
在这里插入图片描述
由上图可以看出:
设某节点下表为i,则其左孩子下表为2i+1,右孩子下表为2i+2。
2.构建大顶堆:调整堆使每个节点大于或等于它的任意孩子
这里我采用的是从下往上调整,经过一次循环可以把最大值调整到堆顶。
在这里插入图片描述
经过一次遍历循环,从下往上调整那么必然可以选出一个最大值放在堆顶,然后去掉第一个把后面的数据再进行递归直到只剩一个节点就结束递归。

void Swap(int* x,int *y)//交换数据
{
    int temp = *x;
    *x = *y;
    *y = temp;
}
void HeapCreate(int *arr, int size)
{
    if(size == 1)
        return;
    int i;
    int n = size%2==0 ? (size-1)/2 : (size-2)/2;//找到最后一个有孩子的节点
    i = n;
    while(i >= 0){
            if(arr[i] < arr[2*i+1]){			//与左孩子比较
                Swap(&arr[i],&arr[2*i+1]);
            }
            //由于循环开始是最后一个有孩子的节点,所以这个节点必有左孩子,但是不一定有右孩子,所以还要判断一下防止数组越界访问
            if(arr[i] < arr[2*i+2] && 2*i+2 < size){//与右孩子比较,
                Swap(&arr[i],&arr[2*i+2]);
            }
            i--;
    }
    HeapCreate(arr+1, size-1);//找出了最大值,把剩下的节点递归
    //其实就是第一遍从下标0开始选出最大值到arr[0],然后再从下标1开始选出次大值到arr[1]......
}

3.个人感受:堆排序和选择排序很类似,只是堆排序在每次调整中已经相对有序了。所以比选择排序少了一些频繁的交换。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值