堆排序-浅见

一直以来都不敢去学习了解堆排序,无形中自己产生对该算法的畏惧心理。今天有幸弄通,特来祭奠!看来世事,只要敢行,就一定行!

概述

堆排序使用数据结构中的堆结构,通过建立堆,然后循环调整堆,完成排序过程。

基础

首先要了解完全二叉树的概念,将堆分为大根堆和小根堆。大根堆用于维护非降序序列(也就是升序排列)。

对于完全二叉树必须记住的一个口诀:

1)父节点i的左子节点位置:2*i + 1

2)父节点i的右子节点位置:2*i + 2

3)子节点i的父节点位置:floor((i - 1) / 2)

上面公式都也不需要死机硬背,只需要知道树结构的父子位置关系是2^h关系即可。

过程

首先,需要创建堆。

void CreateHeap(int A[], int hLen)

{

int begin = hLen >> 2 - 1;

for ( int i = begin; i >= 0; i-- )

{

AdjustHeap(A, hLen, i);

}

}

其次,调整堆。

void AdjustHeap(int A[], int hLen, int i)

{

int left = LeftChildIndex(i);

int right = RightChildIndex(i);

int largest = i;

while (left < hLen && right < hLen)

{

if( left < hLen && A[largest] < A[left] )

{

largest = left;

}

if( right < hLen && A[largest] < A[right] )

{

largest = right;

}

if( largest != i )

{

int temp = A[largest];// 交换最大值与父节点位置

A[largest] = A[i];

A[i] = temp;


i = largest;// 调整父节点位置

left  = LeftChildIndex(i);// 调整子节点位置

right = RightChildIndex(i);

}else{

break;

}

}

}

最后,取调整堆后的堆顶元素,输出排序序列。

void HeapSort(int A[], int aLen)

{

int hLen = aLen;


CreateHeap(A,hLen);

while (hLen > 1)

{

int temp = A[hLen - 1];

A[hLen - 1] = A[0];

A[0] = temp;

hLen --;

AdjustHeap(A,hLen,0);

}

}

具体过程如下图所示,下图为创建一个大根堆过程。

image

堆排序,获取排序序列:

image

注:其中获取左节点、右节点代码为:

int LeftChildIndex(int cur)

{

return (cur << 1) + 1;

}

int RightChildIndex(int cur)

{

return (cur << 1) + 2;

}

int ParentIndex(int cur)

{

return (cur - 1) >> 1;

}

堆排序是一种常见的排序算法,具有稳定性高、效率高等优点。下面介绍一下使用Java实现的小根堆排序。 首先,我们需要定义一个小根堆类,用于存储待排序的数据。该类需要包含以下几个方法: 1. `Heap()`:构造函数,用于初始化小根堆。 2. `insert(int val)`:插入操作,将一个新的元素插入到小根堆中。 3. `deleteMin()`:删除操作,删除小根堆中的最小元素,并返回该元素的值。 4. `size()`:获取小根堆中元素的个数。 5. `isEmpty()`:判断小根堆是否为空。 接下来,我们就可以使用小根堆对待排序的数据进行排序了。具体的步骤如下: 1. 将待排序的数据存入小根堆中。 2. 依次从小根堆中删除最小元素,并将其存入数组中。 3. 最后,将数组反转,即可得到排序后的结果。 下面是具体的Java代码实现: ```java public class HeapSort { public static void heapSort(int[] arr) { Heap heap = new Heap(arr.length); for (int i = 0; i < arr.length; i++) { heap.insert(arr[i]); } for (int i = 0; i < arr.length; i++) { arr[i] = heap.deleteMin(); } // 反转数组 reverse(arr); } // 反转数组 private static void reverse(int[] arr) { int left = 0; int right = arr.length - 1; while (left < right) { int temp = arr[left]; arr[left] = arr[right]; arr[right] = temp; left++; right--; } } // 小根堆类 static class Heap { private int[] heap; private int size; public Heap(int capacity) { heap = new int[capacity + 1]; size = 0; } public void insert(int val) { if (size == heap.length - 1) { throw new RuntimeException("Heap is full"); } int i = ++size; while (i != 1 && val < heap[i / 2]) { heap[i] = heap[i / 2]; i /= 2; } heap[i] = val; } public int deleteMin() { if (isEmpty()) { throw new RuntimeException("Heap is empty"); } int min = heap[1]; int last = heap[size--]; int i = 1; int child = 2; while (child <= size) { if (child < size && heap[child] > heap[child + 1]) { child++; } if (last > heap[child]) { heap[i] = heap[child]; i = child; child *= 2; } else { break; } } heap[i] = last; return min; } public int size() { return size; } public boolean isEmpty() { return size == 0; } } } ``` 使用该算法对数组进行排序: ```java int[] arr = {3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5}; HeapSort.heapSort(arr); System.out.println(Arrays.toString(arr)); // 输出 [1, 1, 2, 3, 3, 4, 5, 5, 5, 6, 9] ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值