哈夫曼树编码-C语言

哈夫曼树编码

1.实验目的

了解二叉树的定义,理解二叉树的基本性质和存储结构,掌握哈夫曼树的构造,实现哈夫曼编码与译码算法。

2.实验内容

从键盘输入一串电文字符与权值,输出对应的哈夫曼编码;从键盘输入一串二进制代码,输出对应的电文字符串。具体步骤如下:

  1. 构造一棵哈夫曼树;
  2. 实现哈夫曼编码;
  3. 对哈夫曼编码生成的二进制串进行译码;
  4. 要求程序中字符和权值是可变的,实现程序的灵活性。
3.实验工具

Dev-C++

4.实验代码
//Authors:xioabei

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
typedef char **HuffmanCode;   //动态分配数组存储哈夫曼编码表 
typedef struct{
 int weight;
 int parent,lchild,rchild;
}HTNode,*HuffmanTree;
typedef struct Code{
 char ch;
 int weight;
 Code *next;
}*CodeLink;
//创建字符与权值 
void CreateCouple(CodeLink &Couple,int n){
 int i;
 CodeLink r,p;
 Couple = (CodeLink)malloc(sizeof(Code));
 Couple->next = NULL;
 r = Couple;
 for(i = 1;i<=n;i++){
  p = (CodeLink)malloc(sizeof(Code));
  printf("[请输入第%d个字符与权值:]\n>>>",i);
  getchar();
  scanf("%c %d",&Couple[i].ch,&Couple[i].weight);
  p->next = NULL;
  r->next = p;
  r = p;
 }
 printf("\n[成功创建字符与权值]\n");
}
//选择最小的两个 
void Select(HuffmanTree HT,int n,int *s1, int *s2){
 int i,min,max;
 for(i=1;i<=n;i++){
  if(HT[i].parent == 0){
   *s2 = i;
   max = i;
   *s1 = i;
  }
 }
 for(i=1;i<=n;i++){
  if(HT[*s1].weight>HT[i].weight && HT[i].parent == 0)
   *s1 = i;
  if(HT[max].weight<HT[i].weight && HT[i].parent == 0)
   max = i;
 }
 min = HT[*s1].weight;
 HT[*s1].weight = HT[max].weight;
 for(i=1;i<=n;i++)
  if(HT[*s2].weight>HT[i].weight && HT[i].parent == 0)
   *s2 = i;
 HT[*s1].weight = min;
 printf("%d--%d\n",HT[*s1].weight,HT[*s2].weight);
}
//创建哈夫曼树 
void CreateHuffmanTree(HuffmanTree &HT,CodeLink Couple,int n){
 printf("\n---------开始创建哈夫曼树---------\n");
 int m,i,s1=1,s2=1,*p = &s1,*q = &s2;
 //构造哈夫曼树HT
 if(n<=1)
  return;
 m=2*n-1;
 HT =  (HTNode*)malloc(sizeof(HTNode)*(m+1)); //由于0号单元未用,所以需要动态分配m+1个单元,HT[m]表示根结点 
 for(i=1;i<=m;i++){       //将1~m号单元初始化为0 
  HT[i].parent = 0;
  HT[i].lchild = 0;
  HT[i].rchild = 0;
 }
 printf("\n初始化成功……\n"); 
 for(i=1;i<=n;i++)       //输入前n个单元叶子结点权值
  HT[i].weight = Couple[i].weight;
//---------- 初始化工作结束,开始创建哈夫曼树 -----------// 
 for(i=n+1;i<=m;i++){
//  通过n-1次选择、删除、合并来创建哈夫曼树
  Select(HT,i-1,p,q);
//  在HT[k](1<=k<=i-1)中选择双亲域为0且权值最小的结点,并且返回它们在HT中序号s1,s2
  HT[s1].parent = i;
  HT[s2].parent = i;
//  得到新结点i,从森林中删除s1,s2,将s1,s2双亲域由0改为i
  HT[i].lchild = s1;      //s1,s2分别作为i的左右孩子 
  HT[i].rchild = s2;
  HT[i].weight = HT[s1].weight + HT[s2].weight; //i的权值为左右孩子权值之和 
 }
 printf("\n---------结束创建哈夫曼树---------\n");
}
//创建哈夫曼编码 
void CreateHuffmanCode(HuffmanTree HT,HuffmanCode &HC,int n){
 int c,f,start,i;
 char *cd; 
 //从叶子结点逆向求每个字符的哈夫曼编码,存储在编码表HC中 
 HC = (HuffmanCode)malloc(sizeof(char*)*(n+1));  //分配n个字符编码表空间 
 cd = (char*)malloc(sizeof(char)*n);    //分配临时存放每个字符编码动态数组空间 
 cd[n-1] = '\0';         //编码结束符 
 for(i=1;i<=n;i++){        //逐个字符求哈夫曼编码 
  start = n-1;   //start开始指向最后,即编码结束位置 
  c = i;
  f = HT[i].parent;  //f指向c的双亲结点
  while(f!=0){   //从叶子结点开始向上回溯,直到根结点 
   --start;   //回溯一次start向前指一个位置 
   if(HT[f].lchild==c)
    cd[start] = '0';//结点c是f的左孩子,则生成代码"0" 
   else
    cd[start] = '1';//结点c是f的右孩子,则生成代码"1" 
   c = f;
   f = HT[f].parent; //继续向上回溯 
  }      //求出第i个字符编码 
  HC[i] = (char*)malloc(sizeof(char)*(n-start)); //为第i个字符编码分配空间 
  strcpy(HC[i],&cd[start]);//将求得的编码从临时空间cd中复制到HC当前行中
  puts(HC[i]);
 }
 free(cd);     //释放临时空间 
 printf("\n---------哈夫曼树编码成功---------\n");
}
//编码 
void Encode(HuffmanCode HC,CodeLink Couple,char T[],int n){
 int i,j,k,t;
 printf("[编码如下:]\n");
 for(i=0;T[i]!='\0';i++)
  for(j=1;j<=n;j++){
   if(Couple[j].ch==T[i]){
    for(k=1;k<=n;k++){
     if(Couple[j].weight==Couple[k].weight)
      puts(HC[k]);
    }
   } 
 }
}
//译码 
void Decode(HuffmanCode HC,CodeLink Couple,char T[],int n){
 int i,j,k,location = 0,len,tag;
 printf("[译码如下:]\n");
 for(i=0;T[location+1]!='\0';i++){
  for(j=1;j<=n;j++){
   len = strlen(HC[j]);
   tag = 1;
   for(k=0;k<len;k++){
    if(HC[j][k]!=T[location+k]){
     tag = 0;
     break;
    }
   }
   if(tag==1){
    printf("%c",Couple[j].ch);
    break;
   }
  }
  len = strlen(HC[j]);
  location += len;
 }
}
// 打印菜单 
void PrintMenu(){
 printf("\n**********菜单**********\n");
 printf("\n1.创建字符与权值;\n");
 printf("2.创建哈夫曼树;\n");
 printf("3.生成哈夫曼编码;\n");
 printf("4.编码;\n");
 printf("5.译码;\n");
 printf("0.退出;\n");
 printf("\n************************\n");
 printf("[请输入你的选择:]\n>>>");
}
//主函数 
int main(){
 HuffmanTree HT;
 HuffmanCode HC;
 CodeLink Couple; 
 int i,n,user;
 char T[100];
 while(1){
  PrintMenu();
  scanf("%d",&user);
  switch(user){
   case 1:{
    printf("[请输入叶子结点数:]\n>>>");
    scanf("%d",&n);
    CreateCouple(Couple,n);
    break;
   }
   case 2:CreateHuffmanTree(HT,Couple,n);break;
   case 3:CreateHuffmanCode(HT,HC,n);break;
   case 4:{
    printf("[请输入要编码的字符:]\n>>>");
    getchar();
    gets(T);
    Encode(HC,Couple,T,n);
    break;
   }
   case 5:{
    printf("[请输入要编码的字符:]\n>>>");
    getchar();
    gets(T);
    Decode(HC,Couple,T,n);
    break; 
   } 
   case 0:exit(0);
  }
 }
 return 0;
}

5.实验结果

Haffuman

Haffuman

6.实验分析

1.HT初态

HT初态
2.HT终态
HT终态
3.示意图
哈夫曼树

7.资料

1951年,哈夫曼在麻省理工学院(MIT)攻读博士学位,他和修读信息论课程的同学得选择是完成学期报告还是期末考试。导师罗伯特·法诺(Robert Fano)出的学期报告题目是:查找最有效的二进制编码。由于无法证明哪个已有编码是最有效的,哈夫曼放弃对已有编码的研究,转向新的探索,最终发现了基于有序频率二叉树编码的想法,并很快证明了这个方法是最有效的。哈夫曼使用自底向上的方法构建二叉树,避免了次优算法香农-范诺编码(Shannon–Fano coding)的最大弊端──自顶向下构建树。

给定N个权值作为N个叶子结点,构造一棵二叉树,若该树的带权路径长度达到最小,称这样的二叉树为最优二叉树,也称为哈夫曼树(Huffman Tree)。哈夫曼树是带权路径长度最短的树,权值较大的结点离根较近。

在计算机数据处理中,哈夫曼编码使用变长编码表对源符号(如文件中的一个字母)进行编码,其中变长编码表是通过一种评估来源符号出现机率的方法得到的,出现机率高的字母使用较短的编码,反之出现机率低的则使用较长的编码,这便使编码之后的字符串的平均长度、期望值降低,从而达到无损压缩数据的目的。

  • 14
    点赞
  • 114
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
哈夫曼树是一种重要的数据结构,它常常被用来进行数据压缩和编码。下面是哈夫曼树的构造及字符编码C语言实现: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #define MAX_NODE_NUM 1000 // 哈夫曼树节点最大数目 #define MAX_CODE_LEN 1000 // 哈夫曼编码最大长度 // 哈夫曼树节点结构体 typedef struct { int weight; // 权重 int parent, lchild, rchild; // 父亲节点、左儿子节点、右儿子节点 } HTNode; // 哈夫曼编码结构体 typedef struct { char ch; // 字符 char code[MAX_CODE_LEN]; // 编码 } HCode; // 建立哈夫曼树,并返回根节点的下标 int CreateHT(HTNode ht[], int n) { int i, j, k, x1, x2, min1, min2; for (i = 0; i < 2 * n - 1; i++) { ht[i].parent = ht[i].lchild = ht[i].rchild = -1; } for (i = n; i < 2 * n - 1; i++) { min1 = min2 = 0x7fffffff; // 设置最大值 x1 = x2 = -1; for (j = 0; j < i; j++) { if (ht[j].parent == -1) { if (ht[j].weight < min1) { min2 = min1; x2 = x1; min1 = ht[j].weight; x1 = j; } else if (ht[j].weight < min2) { min2 = ht[j].weight; x2 = j; } } } ht[x1].parent = i; ht[x2].parent = i; ht[i].lchild = x1; ht[i].rchild = x2; ht[i].weight = ht[x1].weight + ht[x2].weight; } return 2 * n - 2; } // 生成哈夫曼编码 void CreateHCode(HTNode ht[], HCode hcd[], int n) { char cd[MAX_CODE_LEN]; int i, j, c, p; for (i = 0; i < n; i++) { hcd[i].ch = i + 'a'; // 假设字符集为26个小写字母 cd[n - 1] = '\0'; c = i; while (ht[c].parent != -1) { p = ht[c].parent; if (ht[p].lchild == c) { cd[strlen(cd) - 1] = '0'; } else { cd[strlen(cd) - 1] = '1'; } c = p; strcat(cd, " "); } strcpy(hcd[i].code, strrev(cd)); } } int main() { int n, i, root; HTNode ht[MAX_NODE_NUM]; HCode hcd[MAX_NODE_NUM]; printf("请输入字符集的大小:"); scanf("%d", &n); printf("请输入各个字符的权重:"); for (i = 0; i < n; i++) { scanf("%d", &ht[i].weight); } root = CreateHT(ht, n); CreateHCode(ht, hcd, n); printf("字符\t权重\t哈夫曼编码\n"); for (i = 0; i < n; i++) { printf("%c\t%d\t%s\n", hcd[i].ch, ht[i].weight, hcd[i].code); } return 0; } ``` 以上程序中,`CreateHT`函数用于建立哈夫曼树,`CreateHCode`函数用于生成哈夫曼编码。在这个例子中,我们假设字符集为26个小写字母。对于每个字符,我们首先计算它的权重。然后利用`CreateHT`函数建立哈夫曼树,再利用`CreateHCode`函数生成哈夫曼编码。最后输出每个字符的权重和哈夫曼编码。 需要注意的一点是,哈夫曼编码是从叶子节点到根节点的,因此我们需要将最后得到的编码字符串反转。这里我们使用了一个`strrev`函数,它可以将一个字符串反转。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值