哈夫曼编码译码器1要点.docx

上传人:b****3 文档编号:27418123 上传时间:2023-06-30 格式:DOCX 页数:29 大小:68.12KB
下载 相关 举报
哈夫曼编码译码器1要点.docx_第1页
第1页 / 共29页
哈夫曼编码译码器1要点.docx_第2页
第2页 / 共29页
哈夫曼编码译码器1要点.docx_第3页
第3页 / 共29页
哈夫曼编码译码器1要点.docx_第4页
第4页 / 共29页
哈夫曼编码译码器1要点.docx_第5页
第5页 / 共29页
点击查看更多>>
下载资源
资源描述

哈夫曼编码译码器1要点.docx

《哈夫曼编码译码器1要点.docx》由会员分享,可在线阅读,更多相关《哈夫曼编码译码器1要点.docx(29页珍藏版)》请在冰豆网上搜索。

哈夫曼编码译码器1要点.docx

哈夫曼编码译码器1要点

滁州学院

课程设计报告

 

课程名称:

数据结构

设计题目:

哈夫曼编码译码器

系别:

计算机科学与技术

专业:

网络工程

组别:

第十组

起止日期:

2011年4月28日~2011年5月23日

指导教师:

 

计算机科学与技术系2010年制

课程设计题目

哈夫曼编码译码器

组长

学号

班级

系别

计算机科学与技术

专业

网络工程

组员

 

指导教师

课程设计目的

掌握哈夫曼编码译码器的构造以及懂得设计思想

课程设计所需环境

VC++环境

课程设计任务要求

基本了解哈夫曼编码译码器的构造原理

课程设计工作进度计划

序号

起止日期

工作内容

分工情况

郭金华

4月28日

5月23日

收集资料及构建哈夫曼树和打开文件函数

构建哈夫曼树和打开文件函数

朱锟

4月28日

5月23日

收集资料及构建哈夫曼树和打开文件函数

构建哈夫曼树和打开文件函数

刘强

4月28日

5月23日

编辑小结和构建生成哈夫曼编码的函数

构建能够统计文件中字符的函数及其意义

朱开发

4月28日

5月23日

编辑小结和构建生成哈夫曼编码的函数

构建能够统计文件中字符的函数及其意义

高鹏飞

4月28日

5月23日

编辑哈夫曼译码器中的译码器函数

编辑哈夫曼译码器中的的译码器函数

曹探

4月28日

5月23日

编辑哈夫曼译码器中的译码器函数等其他的相关事宜

编辑哈夫曼译码器中的译码器函数等其他的相关事宜

教研室审核意见:

 

教研室主任签字:

年月日

课程设计任务书

目录

1课程设计的目的和意义。

3

2需求分析。

4

3系统(项目)设计。

4

设计思路及方案。

4

模块的设计及介绍。

5

主要模块程序流程图。

7

4系统实现。

7

主调函数。

7

建立HuffmanTree。

8

生成HuffmanTree。

,。

9

电文译码。

10

5小结。

13

6系统调试。

13

参考文献。

13

附录源程序。

13

1.课程设计的目的和意义

随着人民生活水平的提高,科学的进步和通信技术的发展。

人们在进行通信的同时要求做到信息交流的保密性和快速准确性,尤其是在军事和高端科技方面。

为了解决上面所提出的问题、实现该目的,设计出了一种快速可靠的编码方式,即哈夫曼编码。

哈夫曼编码的应用很广泛,利用哈夫曼树求得的用于通信的二进制编码称为哈夫曼编码。

树中从根到每个叶子都有一条路径,对路径的各分支约定:

指制定左子树的分支表示“0”码,指向右子树的分支表示“1”码,去每条路径的“0”或“1”的序列作为和各个对应的字符编码,这就是哈夫曼树编码。

利用哈夫曼树编码进行通信可以大大的提高信道利用率,缩短信息传输时间,降低传输成本。

但是,这要求在发送端通过一个编码系统对传数据预先编码,在接收端将传来数据进行译码(复原)。

通常我们把数据压缩的过程叫做编码,解压缩的过程叫做解码。

电报通信是传递文字的二进制码形式的字符串。

但在信息传递时,总希望总长度尽可能最短,即采用最短码。

作为计算机系学生,我们应该很好的掌握这门技术。

在课堂上,我们能够学到许多的理论知识,但我们很少有过自己动手实践的机会!

课程设计就是为解决这个问题提供了一个平台。

在课程设计过程中,我们每个人选择一个课题,认真研究,根据课堂讲授内容,还可以增强我们独立思考能力和动手能力:

通过编写实验代码和调试运行,我们可以逐步积累调试C程序的经验并逐渐培养我们的编程能力、用计算机解决实际问题能力。

在课程设计过程中,我们不但有自己的独立思考,还借助各种参考文件来帮助我们完成系统。

更为重要的是,我们同学之间加强了交流,在对问题的认识方面可以交换不同的意见。

同时,师生之间的互动也随之改善,我们可以通过具体的实例来从老师那学到更多的知识数据结构课程具有比较强的理论性,同时也具有较强的可用性和实践性。

课程设计是一个重要的教学环节。

我们在一般情况下都能够重视实验环节,但是容易忽略实验的总结,忽略实验报告的撰写。

通过这次试验我们明白:

作为一名大学生必须严格训练分析总结能力、书面表达能力,需要逐步培养书写实验报告以及科技论文的能力。

只有这样,我们的综合素质会有好的提高。

 

2需求分析

课题:

哈夫曼编码译码器系统

问题描述:

打开一篇英文文章,统计该文章中每个字符出现的次数,然后以它作为权值,对每一个字符进行编码,编码完成再对其编码进行译码。

问题补充:

1.从硬盘的一个文件里读出一段英语文章;

2.统计这篇文章中的每个字符出现的次数;

3.以字符出现字数作为权值,构建哈夫曼树,并将哈夫曼树的存储结构的初态和终态进行输出;

4.对每个字符进行编码并将所编码写入文件然后对所编码进行破译。

具体介绍:

在本课题中,我们在硬盘E盘中预先建立一个filel.txt文档,在里面编辑一篇文章(大写)。

然后运行程序,调用fileopen()函数读出该文章,显示在界面;再调用jsq()函数对该文章的字符种类进行统计,并对每个字符的出现次数进行统计,并且在界面上显示;然后以每个字符出现次数作为权值,调用ChuffmanTree()函数构建哈夫曼树;并调用print1()和print2()函数将哈夫曼的存储结构的初态和终态进行输出。

然后调用HuffmanEncoding()函数对哈夫曼树进行编码,调用coding()函数将编码写入文件;再调用decode()对编码进行译码,再输出到界面。

至此,整个工作就完成了。

测试数据:

例如从文本中读到文章为:

IAMASTUDENT。

译码后的字符串:

IAMASTUDENT

Pressanykeytocontinue

 

3系统(项目)设计

(1)设计思路及方案

本课题是用最优二叉树即哈夫曼树来实现哈夫曼编码译码器的功能。

假设每种字符在电文中出现的次数为Wi,编码长度为Li,电文中有n种字符,则电文编码总长度为(W1*L1)+(W2*L2)+.......(Wi*Li)。

若将此对应到二叉树上,Wi为叶结点,Li为根结点到叶结点的路径长度。

那么,(W1*L1)+(W2*L2)+.......(Wi*Li)恰好为二叉树上带权路径长度。

因此,设计电文总长最短的二进制前缀编码,就是以n种字符出现的频率作权,构造一棵哈夫曼树,此构造过程称为哈夫码编码。

该系统将实现以下几大功能:

从硬盘读取字符串,建立哈夫曼树,输出哈夫曼树的存储结构的初态和终态,输出各种字符出现的次数以及哈夫曼编码的译码等。

(2)模块的设计及介绍

从硬盘读取字符串

Fileopen(参数)

{

实现命令;

打印输出;

}

建立HuffmanTree

通过三个函数来实现:

voidselect(参数)

{

初始化;

for{

接受命令;

处理命令;

}

说明:

在ht[l.......k]中选择parent为0且权值最小的两个根结点的算法

intjsq(参数)

{

初始化;

for

{接受命令;

处理命令;

}

}

说明:

统计字符串中各种字母的个数以及字符的种类

voidChuffmanTree()

{初始化;

for

{

接受命令;

处理命令;

}

输出字符统计情况;

}

说明:

构造哈夫曼树

输出哈夫曼树的存储结构的初态和终态

分别调用print1()和print2()来实现

voidprint1(参数)

{

初始化;

输出初态;

}

说明:

输出哈夫曼树的初态

voidprint2(参数)

{

for{

输出终态;

}

}

说明:

输出哈夫曼树的终态

哈夫曼编码和译码

voidHuffmanEncoding(参数)

{

定义变量;

{

处理命令;

}

}

说明:

哈夫曼编码

char*decode(参数)

{

定义变量;

while{

接受命令;

处理命令;

}

}

说明:

哈夫曼译码

 

哈夫曼编码:

流程图解释:

该流程图表示哈夫曼编码情况。

首先初始化,Cd[--start]=0,start=num.然后进行编码,使用了一个三目运算符。

Cd[--start]=(T[p].lchild==c)?

’0’:

’1’,即当cd[--start]=T[p].lchild==c时,cd[--start]=0;当cd[--start]=0;当cd[--start]=T[p].lchild!

=c时,cd[--start]=1.这个编码循环一直到i=num时结束。

 

4系统实现

各模块关键代码及算法的解释:

主调函数

代码解释:

这是main函数里的各个函数调用情况。

fileopen(string);//从C盘内中读取文件

num=jsq(string,cnt,str);//统计字符种类及各类字符出现的频率

DhuffmanTree(HT,cnt,str);

printf(“HuffmanTree的初态:

\n”);

print1(HT);//输出哈夫曼树的初态

ChuffmanTree(HT,HC,cnt,str);//建立哈夫曼树

HuffmanEncoding(HT,HC);//生成哈夫曼编码

printf(“HuffmanTree的终态:

\n”);

print2(“HuffmanTree的终态:

\n”);

printf(“%s\n”);//输出译码后的字符串

建立HuffmanTree

代码解释:

该函数为在ht[l......k]中选择patent为0且权值最小的根结点的算法,其序号为s1和s2.

voidselect(HuffmanTreeT,intk,int&s1,int&s2)

{

inti,j;

s2=s1

for(i=1;i<=k;i++)

if(T[i].weight

{

j=i;min1=T[i].weight;

}

s1=j;min1=32767;

for(i=1;i<=k;i++)

if(T[i].weight

=s1)

{

j=i;min1=T[i].weight;

}

s2=j;

}

对所有的字母进行统计种类和出现的次数

intjsp(char*s,intcnt[],charstr[])

{

inti,j,k;

char*p;

inttemp[27];

for(i=1;i<=26;i++)

temp[i]=0;

for(p=s;*p!

=’\0’;p++)

{

if(*p>=’A’&&*p<=’Z’)

k=*p-64;//找到该字母的位置

temp[k]++;

}//统计各种字符的个数

for(i=1,j=0;i<=26;++i)

if(temp[i]!

=0)

{

j++;

str[j]=i+64;//送对应字母到数组中

cnt[j]=temp[i];//存入对应字母的权值

}

returnj;//j是输入字母总数

}

代码解释:

下面函数用来构造哈夫曼树HT。

首先初始化哈夫曼树,然后输入前面统计的各个结点的权值,用for循环来构造哈夫曼树。

voidChuffmanTree(HuffmanTreeHT,HuffmanCodeHC,intcnt[],charstr[])

{

inti,s1,s2;

for(i=1;i<=2*num-1;i++)//初始化HT,2*num-1是指哈夫曼所有的节点数目

{

HT[i].lchild=0;HT[i].rchild=0;

HT[i].patent=0;HT[i].weight=0;

}

for(i=1;i<=num;i++)//输入num个叶结点的权值

HT[i].weight=cnt[i];

for(i=num+1;i<=2*num-1;i++)

{

select(HT,i-1,s1,s2);

HT[s1].patent=i;HT[s2].patent=i;

HT[i].lchild=s1;HT[i].rchild=s2;

HT[i]weight=HT[s1].weight*HT[s2].weight;

}//在ht[l.....k]中选择patent为0且权值最小的两个根结点,其序号为s1和s2,i为双亲

for(i=0;i<=num;i++);//输入字符集中字符

HC[i].ch=str[i];//字符种类

i=1;while(i<=num)

printf(“字符%c次数:

%d”,HC[i].ch,cnt[i++]);

}//输出统计的情况

生成Huffman编码并写入文件

代码解释:

根据哈夫曼树T求哈夫曼编码H。

voidHuffmanEncoding(HuffmanT\reeT,HuffmanCodeH){

intc,p,i;//c和p分别指示t中孩子和双亲

charcd[n];//临时存放编码串

intstart;//指示码在cd中的起始位置

cd[num]=’\0’;//最后一位(第num个)放上串结束符

for(i=1;i<=num;++i){

start=num;//初始位置

c=i;//从叶子结点t[i]开始上溯

while((p=T[c].patent)>0)//直至上溯到t[c]是树根为止

{

Cd[--start]=(T[p].lchild==c)?

’0’:

’1’;

c=p;

}//若t[c]是t[p]的左孩子则生成0;否则生成1

Strcpy(H[i].bits,*cde[start]);

H[i].len=num-start;

}

}

代码解释:

对str所代表的字符串进行编码并写入文件。

将翻译的二进制码写入文本文件。

voidcoding(HuffmanCodeHC,char*str){

inti,j;

FILE*fp;

Fp=fopen(“codefile.txt”,”w”);

while(*str){

for(i=1;i<=num;i++)

if(HC[i].ch==*str){

for(j=0;j<=HC[i].len;j++)

Fputc(HC[I].bits[j],fp);

break;

}

str++;

}fclose(fp);

}

 

电文译码

代码解释:

代码文件codefile。

Txt的译码,将翻译的二进制码译成原来的字符。

char*decode(HuffmanCodeHC){

FILE*fp;

charstr[254];//假设原文本文件不超过254个字符

char*p;

staticcharcd[n+1];

inti,j,k=0,cjs;

Fp=fopen(“codefile.txt”,”r”);//一只读的方式打开文本文档codefile。

Txt

while(!

feof(fp))//feof(fp)判断文件是否真正结束,feof(fp)=1时文件结束

{

cjs=0;

For(i=0;i

feof(fp);i++){

Cd[i]=’’

Cd[i+1]=’\0’;

Cd[i]=fgetc(fp);//数组接收从fp指针所指向文件中读入的一个字符

for(j=1;j<=num;j++)

if(strcmp(HC[j].bits,cd)==0){

str[k]=HC[j].ch;

k++;

cjs=1;break;

}//haffman编码和密码译码相比较

str[k]=’\0’;

p=str;

returnp;

}

 

5系统调试

本次测试是在我的电脑的E盘中建立一个名为file1.txt的文本文档,其中有大写字母IAMASTUDENT,期望程序能将其读出至界面并实现其它相关的功能。

运行程序后,我们可以见到以下运行界面。

输出哈夫曼树存储结构的初态

输出哈夫曼树存储结构的终态

 

输出译码后的字符

由此可见,此次测试很成功。

我们能够将文本文档file1.txt中的文段读出,并将其统计输出字符种类和每种字符出现的频率。

同时输出哈夫曼树存储结构的初态和终态。

然后输出译码后的字符。

 

小结

-通过两周的课程设计是我对哈夫曼树以及哈夫曼编码有了更深的认识和理解,也使我更加明白了哈夫曼编码译码在信息技术中的重要性地位。

首先我谈谈我在设计期间我遇到的难点。

开始的时候,代码中有许多的错误,特别是有一个“无法找到文件”的错误让我束手无策,最后还是屏蔽了定义的四个头文件然后慢慢的改正错误才让我又看到了希望。

然后在实现文章的读入时,由于对文件不是太熟悉,只好翻开C语言书本仿照其模式编写,但后来进入了死循环,最后的解决方式是把main函数里的一个dowhile循环去掉,在程序中,我还另外加了一个功能-----输出哈夫曼树的存储结构的初态和终态。

这使得我更加的明白了哈夫曼到底是怎么存储信息的。

许多的错误让我明白了一个道理-----细心是非常重要的。

同时,对于编程者而言,思路清晰是相当重要的。

在适当的时候和同学一起交流探讨是一个十分好的学习机会。

请教老师也很重要,因为毕竟我们是新手,对于某些问题很难弄清楚。

而且,某些错误对于我们来说有时候想半天都弄不来,但老师几下就搞好了,这样就更加有效的节约了时间。

这段时间里,可谓是酸甜苦辣都尝过。

有时,为了一个错误而焦头烂额;有时,变成熬夜到凌晨两点;有时,连吃饭都是草草了事;但一旦程序运行成功,总会让我兴奋不已。

通过这次课程设计,我看清楚了自己的编程功底和动手能力还不如人意,这主要是平时是实践太少的缘故。

我想,在未来的暑假中,首先任务是编写一些程序。

虽然我们网络工程专业的学生,但现在变成的能力太差了,毕竟多精通一门技术总是好事。

在这个程序中,还有许多地方值得完善。

比如,读出文本只能是大写的文档,空格和小写不能识别,更不用说是任意的ASCII码字符了。

由于时间问题,暂时不能实现了。

 

参考文献

【1】严蔚敏,数据结构(C语言版),清华大学出版社,2007

【2】苏仕华,数据结构课程设计,机械工业出版社,2007

【3】谭浩强,C语言程序设计教程,高等教育出版社,2006

附录:

源程序

#include

#include

#include

#include

//*********************类型相关变量的定义****************************

#definen100//叶子结点数

#definem2*n-1//哈夫曼树中的结点数

typedefstruct{

charch;//相关的字母

charbits[9];//存放编码位串

intlen;//该字母编码的长度

}CodeNode;//编码的类型

typedefCodeNodeHuffmanCode[n+1];//所有的叶子结点的编码数组

typedefstruct{

intweight;//权值

intlchild,rchild,parent;//左右孩子及双亲指针

}HTNode;//哈夫曼树结点的类型

typedefHTNodeHuffmanTree[m+1];//0号单元不用

intnum;//统计每种字母出现的次数和种类数目

///////////////////////////////建立HuffmanTree////////////////////

voidselect(HuffmanTreeT,intk,int&s1,int&s2){

//在ht[1......k]中选择parent为0且权值最小的两个根结点的算法

//其序号为s1和s2

inti,j;intmin1=101;//min1的数字无何意义只是初始值之后用来记录权值,i为循//环

//最小权值的下标,k为数组结点的总数

for(i=1;i<=k;i++)

if(T[i].weight

j=i;min1=T[i].weight;//赋权值

}

s1=j;min1=32767;//找到权值最小的其中之一

for(i=1;i<=k;i++)

if(T[i].weight

=s1)//不让s2和s1的数组下标重合

{

j=i;min1=T[i].weight;//找到权值最小的其中之一

}

s2=j;

}

intjsq(char*s,intcnt[],charstr[])//str[]存放所有字符,cnt[]来存放每种字母的权值

{//统计字符串中各种字母的个数以及字符的种类,s为数组

//的首地址指针

inti,j,k;

char*p;

intte

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 高等教育 > 院校资料

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1