- 论坛徽章:
- 0
|
本帖最后由 jack1007 于 2012-10-25 10:04 编辑
堆排序思想:
首先把数组看成是一个无序的堆结构,然后首先把数组构建成大根堆,及所有父节点不小于他的两个子节点。
构建的大根堆的特点是顶点值一定不小于所有子节点的值。
然后把大根堆的顶点,也就是数组第一个元素值和堆的尾节点值交换。
这样最大值就一定最后面。尾节点之前的堆结构就失去大根堆特性了,然后需要重新构建大根堆(除了尾节点以外,因为尾节点已经有顺序了,为最大值)
然后再次交换顶点和倒数第二个堆节点的值,然后再重构大根堆(除堆结尾已经交换过的值)。
重复上面步骤,直到只有最顶层的堆节点,这时候整个堆结构为一个递增的数组序列(虽然堆已经不是大根堆了)。
整个算法的核心就是构建大根堆(heap_ex函数实现),其中要注意堆节点和数组键,堆节点从1开始,数组键从0开始。
整个堆排序代码如下:- #include <stdio.h>
- void swap(int *a, int *b);
- void print_arr(int *arr, int arr_len);
- void heap_build(int *arr, int note, int arr_len);
- void heap_sort(int *arr, int note, int arr_len);
- void heap_ex(int *arr, int i, int arr_end);
- int
- main()
- {
- int arr[] = {34, 12, 77, 85, 98, 42, 33, 18, 64, 59, 83};
- int arr_len;
- int note = 0;
- arr_len = sizeof(arr) / sizeof(arr[0]);
- print_arr(arr, arr_len); //打印未排序前的数组
- heap_build(arr, note, arr_len - 1);
- print_arr(arr, arr_len); //打印第一次构建大根堆后的数组
- heap_sort(arr, note, arr_len - 1);
- print_arr(arr, arr_len); //打印排序后的数组
- }
- void
- heap_build(int *arr, int i_note, int arr_end)
- {
- //build max_heap, the note i is the heap note
- int i;
- int arr_len = arr_end + 1;
- for (i = arr_len / 2; i > 0; i-- ) {
- heap_ex(arr, i, arr_end); //move heap_note func
- }
- }
- void
- heap_ex(int *arr, int i, int arr_end)
- {
- //the core func in this program, note_i is the key of the arr
- int note_i = i - 1;
- int child_i = 2 * note_i + 1;
- while (child_i <= arr_end) {
- if (child_i + 1 <= arr_end && arr[child_i] < arr[child_i + 1])
- child_i++;
- if (child_i <= arr_end && arr[child_i] > arr[note_i]) {
- swap(&arr[child_i], &arr[note_i]);
- note_i = child_i;
- child_i = 2 * note_i + 1;
- } else
- break;
- }
- }
- void
- heap_sort(int *arr, int note, int arr_end)
- {
- /*
- *the first value swap with the end value,
- *than rebuild the max_heap except the end of the values which have been swaped with the first value
- */
- int i = arr_end;
- while (i >= 0) {
- swap(&arr[0], &arr[i]);
- i--;
- heap_build(arr, note, i);
- }
- }
- void
- swap(int *a, int *b)
- {
- int tmp;
- tmp = *a;
- *a = *b;
- *b = tmp;
- }
- void
- print_arr(int *arr, int arr_len)
- {
- int i = 0;
- while (i < arr_len) {
- printf("%d ", arr[i]);
- i++;
- }
- printf("\n");
- }
复制代码 |
|