哈夫曼树实验报告

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

哈夫曼树实验报告
一、问题描述
利用哈夫曼编码进行通信可以大大提高信道利用率,缩短信息传输时间,降低传输成本。

但是,这要求在发送端通过一个编码系统对待传数据预先编码,在接收端将穿来的数据进行译码,此试验即设计这样的一个简单的编/译码系统。

系统应该具有如下的几个功能。

1. 接受原始数据
从终端任意读入字母,求出其各自的权重值,建立哈夫曼树,并将它存于hfmtree.dat文
件中。

2. 编码
利用已建好的哈夫曼树,对文件中的正文进行编码,然后将结果存入codefile.dat中。

3. 译码
利用已建好的哈夫曼树将文件codefile.dat中的代码进行译码,结果存入文件textfile.dat
中。

4. 打印编码规则
即字符与编码的一一对应关系。

5. 打印哈夫曼树
将已存在内存中的哈夫曼树以直观的方式显示在终端上。

二、数据结构设计
1. 构造哈夫曼树时使用静态量表作为哈夫曼树的存储。

在构造哈夫曼树时,设计一个结构体数组HuffNode保存哈夫曼树中各节点的信息,
根据二叉树的性质可知,具有n个叶子节点的哈夫曼树共有2n-1个结点,所以数组
HuffNode的大小设置为2n-1,描述节点的数据类型为:
typedef struct
{
int weight; //结点权值
int parent;
int lchild;
int rchild;
}HNodeType;
2. 求哈夫曼编码时使用一位结构数组HuffCode作为哈腹满编码信息的存储。

求哈夫曼编码,实质上就是在以建立的哈夫曼树中,从叶子结点开始,沿结点的双亲链域退到根结点,每退回一步,就走过了哈夫满树的一个分支,从而得到一位哈夫曼码值,由于一个字符的哈夫曼编码是从根结点到相应叶子结点所经过的路径上各分支所组成的0、1序列,因此先得到的分支代码为所求编码的低位码,后得到的分支代码为所求编码的高位码,所以设计如下数据类型:
typedef struct
{
int bit[26];
int start;
}HCodeType;
3. 文件hfmtree.dat、codefile.dat、和textfile.dat。

三、代码实现
#include<iostream>
#include<fstream>
using namespace std; #define Maxvalu 100 typedef struct {
int weight;
int parent;
int lchild;
int rchild;
}HNodeType;
typedef struct
{
int w;
char x;
}CNode;
typedef struct
{
int bit[26];
int start;
}HCodeType;
void HuffmanTree(int n,CNode c[],HNodeType *HuffNode[]) {
int i,m1,m2,x1,x2,j;
for(i=0;i<n;i++)
{
HuffNode[i]->weight=c[i].w;
}
for(i=0;i<n-1;i++)
{
m1=m2=Maxvalu;
x1=x2=0;
for(j=0;j<n+i;j++)
{
if(HuffNode[j]->parent==-1&&HuffNode[j]->weight<m1)
{
m2=m1;
x2=x1;
m1=HuffNode[j]->weight;
x1=j;
}
else
if(HuffNode[j]->parent==-1&&HuffNode[j]->weight<m2)
{
m2=HuffNode[j]->weight;
x2=j;
}
}
HuffNode[x1]->parent=n+i;
HuffNode[x2]->parent=n+i;
HuffNode[n+i]->weight=HuffNode[x1]->weight+HuffNode[x2]->weight;
HuffNode[n+i]->lchild=x1;
HuffNode[n+i]->rchild=x2;
}
cout<<"哈夫曼二叉树为:"<<endl;
for(i=0;i<2*n-1;i++)
{
cout<<HuffNode[i]->weight<<" "<<HuffNode[i]->lchild<<" "<<HuffNode[i]->rchild<<" "<<HuffNode[i]->parent<<endl; }
}
void HaffmanCode(int n,CNode c[],HNodeType *HuffNode[]) {
HCodeType HuffCode[26],cd;
int i,a,p,j,q=2*(n-1);
ofstream outfile("file3.txt",ios::out);
if(!outfile)
{
cerr<<"打开文件file3.txt失败~"<<endl;
exit(0);
}
for(i=0;i<n;i++)
{
cd.start=n-1;
a=i;
p=HuffNode[a]->parent;
while(p!=-1)
{
if(HuffNode[p]->lchild==a)
cd.bit[cd.start]=0;
else
cd.bit[cd.start]=1;
cd.start--;
a=p;
p=HuffNode[a]->parent;
}
for(j=cd.start+1;j<n;j++) HuffCode[i].bit[j]=cd.bit[j]; HuffCode[i].start=cd.start;
}
outfile<<"哈夫曼编码为:"<<endl;
for(i=0;i<n;i++)
{
outfile<<c[i].x<<": ";
for(j=HuffCode[i].start+1;j<n;j++) {
outfile<<HuffCode[i].bit[j];
}
outfile<<endl;
}
outfile.close();
}
void HaffmanDecode(int n,CNode c[],HNodeType *HuffNode[]) {
int i,j,p,a[50],m;
ofstream outfile("file4.txt",ios::out);
if(!outfile)
{
cerr<<"打开文件file4.txt失败~"<<endl;
exit(0);
}
cout<<"请输入哈夫曼译码的代号:"<<endl;
i=0;
cin>>a[i];
while(a[i]==0||a[i]==1)
{
i++;
cin>>a[i];
}
m=i;
outfile<<"输入的代号为:"<<endl;
for(i=0;i<m;i++)
{
outfile<<a[i]<<" ";
}
outfile<<endl;
outfile<<"哈夫曼译码为:"<<endl;
i=0;
for(j=0;a[i]==0||a[i]==1;j++)
{
p=2*(n-1);
while((HuffNode[p]->lchild!=-1||HuffNode[p]->rchild!=-1)&&i<m) {
if(a[i]==0)
p=HuffNode[p]->lchild;
else
if(a[i]==1)
p=HuffNode[p]->rchild;
i++;
}
outfile<<c[p].x<<endl;
}
if(i==m&&(HuffNode[p]->lchild!=-1||HuffNode[p]->rchild!=-1)) {
outfile<<"无对应译码,不能继续译码!"<<endl;
}
outfile.close();
}
int main()
{
CNode c[26];
char ch[100],ch1;
int i,j,m;
int k=0,n=0,a=0;
HNodeType *HuffNode[51];
ofstream outfile("file2.txt",ios::out);
if(!outfile)
{
cerr<<"打开文件file2.txt失败~"<<endl;
exit(0);
}
for(i=0;i<26;i++)
{
c[i].x=NULL;
c[i].w=0;
}
cout<<"请输入一串字母并以“#”号结束。

"<<endl; j=0;
cin>>ch[j];
while(ch[j]!='#')
{
j++;
cin>>ch[j];
}
m=j;
ch1=ch[0];
outfile<<"输入的字母为:"<<endl; for(j=0;j<m;j++)
outfile<<ch[j];
outfile<<endl;
n=0;
j=0;
c[n].x=ch[j];
c[n].w++;
while(ch[j]!='#')
{
j++;
for(i=0;i<n+1;i++)
{
if(ch[j]==c[i].x)
{
c[i].w++;
break;
}
}
if(i==n+1)
{
n++;
c[n].x=ch[j];
c[n].w++;
}
}
for(i=0;i<n;i++)
{
outfile<<c[i].x<<"---权值是:"<<c[i].w<<endl; }
for(i=0;i<2*n-1;i++)
{
HuffNode[i]=new HNodeType;
HuffNode[i]->weight=0;
HuffNode[i]->parent=-1;
HuffNode[i]->lchild=-1;
HuffNode[i]->rchild=-1;
}
HuffmanTree(n,c,HuffNode);
HaffmanCode(n,c,HuffNode);
HaffmanDecode(n,c,HuffNode);
return 0;
outfile.close();
}。

相关文档
最新文档