免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 5813 | 回复: 0
打印 上一主题 下一主题

[算法] C语言堆排序算法实现 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-10-25 10:02 |只看该作者 |倒序浏览
本帖最后由 jack1007 于 2012-10-25 10:04 编辑

堆排序思想:
首先把数组看成是一个无序的堆结构,然后首先把数组构建成大根堆,及所有父节点不小于他的两个子节点。
构建的大根堆的特点是顶点值一定不小于所有子节点的值。
然后把大根堆的顶点,也就是数组第一个元素值和堆的尾节点值交换。
这样最大值就一定最后面。尾节点之前的堆结构就失去大根堆特性了,然后需要重新构建大根堆(除了尾节点以外,因为尾节点已经有顺序了,为最大值)
然后再次交换顶点和倒数第二个堆节点的值,然后再重构大根堆(除堆结尾已经交换过的值)。
重复上面步骤,直到只有最顶层的堆节点,这时候整个堆结构为一个递增的数组序列(虽然堆已经不是大根堆了)。
整个算法的核心就是构建大根堆(heap_ex函数实现),其中要注意堆节点和数组键,堆节点从1开始,数组键从0开始。
整个堆排序代码如下:
  1. #include <stdio.h>

  2. void swap(int *a, int *b);
  3. void print_arr(int *arr, int arr_len);
  4. void heap_build(int *arr, int note, int arr_len);
  5. void heap_sort(int *arr, int note, int arr_len);
  6. void heap_ex(int *arr, int i, int arr_end);

  7. int
  8. main()
  9. {
  10.   int arr[] = {34, 12, 77, 85, 98, 42, 33, 18, 64, 59, 83};
  11.   int arr_len;
  12.   int note = 0;
  13.   arr_len = sizeof(arr) / sizeof(arr[0]);
  14.   print_arr(arr, arr_len);  //打印未排序前的数组
  15.   heap_build(arr, note, arr_len - 1);
  16.   print_arr(arr, arr_len);  //打印第一次构建大根堆后的数组
  17.   heap_sort(arr, note, arr_len - 1);
  18.   print_arr(arr, arr_len);  //打印排序后的数组
  19. }

  20. void
  21. heap_build(int *arr, int i_note, int arr_end)
  22. {
  23.   //build max_heap, the note i is the heap note
  24.   int i;
  25.   int arr_len = arr_end + 1;
  26.   for (i = arr_len / 2; i > 0; i-- ) {
  27.     heap_ex(arr, i, arr_end);  //move heap_note func
  28.   }
  29. }

  30. void
  31. heap_ex(int *arr, int i, int arr_end)
  32. {
  33.   //the core func in this program, note_i is the key of the arr
  34.   int note_i = i - 1;
  35.   int child_i = 2 * note_i + 1;
  36.   while (child_i <= arr_end) {
  37.     if (child_i + 1 <= arr_end && arr[child_i] < arr[child_i + 1])
  38.       child_i++;
  39.     if (child_i <= arr_end && arr[child_i] > arr[note_i]) {
  40.       swap(&arr[child_i], &arr[note_i]);
  41.       note_i = child_i;
  42.       child_i = 2 * note_i + 1;
  43.     } else
  44.       break;
  45.   }
  46. }

  47. void
  48. heap_sort(int *arr, int note, int arr_end)
  49. {
  50.   /*
  51.   *the first value swap with the end value,
  52.   *than rebuild the max_heap except the end of the values which have been swaped with the first value
  53.   */
  54.   int i = arr_end;
  55.   while (i >= 0) {
  56.     swap(&arr[0], &arr[i]);
  57.     i--;
  58.     heap_build(arr, note, i);
  59.   }
  60. }


  61. void
  62. swap(int *a, int *b)
  63. {
  64.   int tmp;
  65.   tmp = *a;
  66.   *a = *b;
  67.   *b = tmp;
  68. }

  69. void
  70. print_arr(int *arr, int arr_len)
  71. {
  72.   int i = 0;
  73.   while (i < arr_len) {
  74.     printf("%d ", arr[i]);
  75.     i++;
  76.   }
  77.   printf("\n");
  78. }
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP