当前位置:   article > 正文

数据结构——哈夫曼树与哈夫曼编码_哈夫曼树与哈夫曼编码编程实现

哈夫曼树与哈夫曼编码编程实现

1. 哈夫曼树

1.1 基本概念

路径:指从根结点到该结点的分支序列。
路径长度:指根结点到该结点所经过的分支数目。
结点的带权路径长度:从树根到某一结点的路径长度与该结点的权的乘积。
树的带权路径长度(WPL):树中从根到所有叶子结点的各个带权路径长度之和。

哈夫曼树是由 n 个带权叶子结点构成的所有二叉树中带权路径长度最短的二叉树,又称最优二叉树。如上图中第三棵树就是一棵哈夫曼树

1.2 构造哈夫曼树
构造哈夫曼树的算法步骤:
① 初始化:用给定的 n 个权值{w1,w2,…,wn}构造 n 棵二叉树并构成的森林F={T1,T2,…,Tn},其中每一棵二叉树Ti(1<=i<=n)都只有一个权值为 wi 的根结点,其左、右子树为空。
② 找最小树:在森林 F 中选择两棵根结点权值最小的二叉树,作为一棵新二叉树的左、右子树,标记新二叉树的根结点权值为其左、右子树的根结点权值之和。
③ 删除与加入:从 F 中删除被选中的那两棵二叉树,同时把新构成的二叉树加入到森林 F 中。
④ 判断:重复②、③操作,直到森林中只含有一棵二叉树为止,此时得到的这棵二叉树就是哈夫曼树。
简单的说就是先选择权小的,所以权小的结点被放置在树的较深层,而权较大的离根较近,这样一来所构成的哈夫曼树就具有最小带权路径长度。

2. 哈夫曼编码实现
2.1 哈夫曼编码

对一棵具有n个叶子结点的哈夫曼树,若对树中的每个左分支赋0,右分支赋1(或左1右0),则从根到每个叶子的通路上,各个分支的赋值分别构成一个二进制串,该二进制串就称为哈夫曼编码。哈夫曼编码是最优前缀编码,能使各种报文对应的二进制串的平均长度最短。
代码如下:

  1. #include<stdio.h>
  2. #include<string.h>
  3. #include<malloc.h>
  4. typedef struct hnode
  5. { int weight;
  6. int lchild,rchild,parent;
  7. }HTNode,*HuffmanTree;/*定义二叉树的存储结点*/
  8. typedef char **HuffmanCode;
  9. void Select(HTNode HT[],int len,int &s1,int &s2)//选出权值最小的两个结点,下标通过s1和s2传出去
  10. {
  11. int i,min1=32767,min2=32767;
  12. for(i=1;i<=len;i++)
  13. {
  14. if(HT[i].weight<min1&&HT[i].parent==0)
  15. {
  16. s2=s1;
  17. min2=min1;
  18. min1=HT[i].weight;
  19. s1=i;
  20. }
  21. else if(HT[i].weight<min2&&HT[i].parent==0)
  22. { min2=HT[i].weight;
  23. s2=i;
  24. }
  25. }
  26. }
  27. void CreateHuffman_tree(HuffmanTree &HT,int n)/*建立哈夫曼树*/
  28. {
  29. int s1, s2;
  30. if(n <= 1)
  31. {
  32. return ;
  33. }
  34. int m = 2*n - 1;
  35. HT = new HTNode[m + 1];
  36. int i = 0;
  37. for(i = 1; i <= m; i++)
  38. {
  39. HT[i].parent = 0;
  40. HT[i].lchild = HT[i].rchild = 0;
  41. }
  42. printf("哈夫曼树各节点的值:");
  43. for(i = 1; i <= n; i++)
  44. {
  45. scanf("%d", &HT[i].weight);
  46. }
  47. for(i = n + 1; i <=m; i++ )
  48. {
  49. Select(HT, i-1, s1, s2);
  50. HT[s1].parent = HT[s2].parent = i;
  51. HT[i].lchild = s1;
  52. HT[i].rchild = s2;
  53. HT[i].weight = HT[s1].weight + HT[s2].weight;
  54. }
  55. }
  56. void Huffman_code(HuffmanTree HT,HuffmanCode &HC,int n)/*哈夫曼树编码*/
  57. {
  58. char *cd;
  59. int start = 0;
  60. int c = 0, f = 0, i = 0;
  61. HC = new char*[n + 1];
  62. cd = new char[n];
  63. cd[n-1] = '\0';
  64. for(i = 1; i <= n; ++i)
  65. {
  66. start = n - 1;
  67. c = i;
  68. f = HT[i].parent;
  69. while(f!= 0)
  70. {
  71. --start;
  72. if(HT[f].lchild == c)
  73. {
  74. cd[start] = '0';
  75. }
  76. else
  77. {
  78. cd[start] = '1';
  79. }
  80. c = f;
  81. f = HT[f].parent;
  82. }
  83. HC[i] = new char[n-start];
  84. strcpy(HC[i], &cd[start]);
  85. }
  86. delete cd;
  87. }
  88. int main()
  89. {
  90. HuffmanTree HT;
  91. HuffmanCode HC;
  92. int i, n;
  93. printf("哈夫曼树节点个数:");
  94. scanf("%d",&n);
  95. CreateHuffman_tree(HT, n);/*建立哈夫曼树*/
  96. Huffman_code(HT,HC,n);/*哈夫曼树编码*/
  97. for(i=1;i<=n;i++)/*输出字符、权值及编码*/
  98. printf("编码是:%s\n",HC[i]);
  99. return 0;
  100. }

结果如下:

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/菜鸟追梦旅行/article/detail/654566
推荐阅读
相关标签
  

闽ICP备14008679号