ImageVerifierCode 换一换
格式:DOCX , 页数:13 ,大小:214.21KB ,
资源ID:6259832      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/6259832.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(哈夫曼编码与译码报告要点.docx)为本站会员(b****6)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

哈夫曼编码与译码报告要点.docx

1、哈夫曼编码与译码报告要点一、设计思想程序要求:利用哈夫曼树对字符串进行编码,要求每个字符有自己唯一的编码。将得到的一串字串译成0、1编码后存到一个文件夹中,然后再从这个文件夹中读出这串编码进行解码。实现方法:输入一串字符,要求字符的区间为大写的26个英文字母, 将获得的串字符用计算权值的函数(jsquanzhi())进行字符统计,统计出现的字符种数以及每种字符出现的次数, 将该种字符出现的次数作为它的权值。将出现的字符的权值和该字符依次分别赋给两个结构体HT和HC,利用HT(节点)权值的大小建立哈夫曼树,首先用选择函数select()函数选择两个权值最小的字符作为叶子节点,创建一个新的节点作为

2、这两个叶节点的父节点,被选中的节点给他的HTi.parent赋值是他下次不再被选中,父节点的权值为,子节点的权值之和。然后将该将父节点放入筛选区中,再进行选择(被选过的不再被使用),直到所有的节点都被使用,这样一个哈夫曼树就被建立了。根据每个字符在哈夫曼书中的位置来编译每个字符的0、1密文代码,从叶节点判断该叶节点是其父节点的左右字左字为0,右子为1,在判断父节点是上级父节点的左右子直至根节点,将生成的0、1字符串按所表示的字符倒序存入HC相应的字符的bins数组。重新一个一个字符的读取输入的字符串,按照字符出现的顺序将它转为0、1代码并存到一个txt文件夹中去。解码时从文件夹中,一个一个字符

3、的读出那串0、1代码赋给一个临时字符串cd,用该字符串与每个字符的HCi.bins密文比较,直到与一个字符的密文相同时,译出该字符,将字符存放在临时字符数组tempstr中,清空临时字符串继续读取0、1代码进行翻译,直至文件密文结束为止。于是就得到了原先被加密的那串字符串。二、算法流程图三、源代码/ hafuman.cpp : 定义控制台应用程序的入口点。/#include stdafx.h#include stdlib.h#include string.h#include stdio.h#define n 100 /叶节点的个数小等于n#define m 2*n-1 /总结点的个数为m=2*

4、n-1int num; /定义一个全局变量用于存放字符种类个数typedef struct /结构体用于存放树节点包括节点的父节点、左子、右子以及权值 int weight;int parent,lchild,rchild;HTNode;typedef HTNode HafumanTreem+1; /重命名HTNodetypedef struct /结构体由于存放每个字符的密文和长度char ch;char bits10;int len;CodeNode;typedef CodeNode HafumanCoden+1; /重命名CodeNodeint _tmain(int argc, _TCH

5、AR* argv)int quan27; /声明一个数组用以存放26字符的权值 char getstr300,str27; /声明两个字符串数组一个用于存输入一个由于存放输入中含有的字符char *s; /声明一个char型指针用于指向字符HafumanTree HT; /声明m+1个树节点HafumanCode HC; /声明n+1个codeint jisuanquan(char *s,int quan,char str); /声明需要调用的函数void gjhafumantree(HafumanTree HT,HafumanCode HC,int quan,char str);void H

6、afumanencode(HafumanTree HT,HafumanCode HC);void coding(HafumanCode HC,char *str);char *decode(HafumanCode HC);printf(请输入要加密的字符串:n);gets(getstr); /获得输入的字符串num=jisuanquan(getstr,quan,str); /统计字符串中含有字符种类个数/printf(%dn,num);gjhafumantree(HT,HC,quan,str); /根据字符权值构建哈夫曼树 Hafumanencode(HT,HC); /根据哈夫曼树确定每个字符

7、的codecoding(HC,getstr); /将字符串译码存入文件夹s=decode(HC); /将暗文解码 printf(解密为:n);printf(%sn,s);system(pause);return 0;/函数int jisuanquan(char *s,int quan,char str) /计算字符串中字符权值char *p; int i,j,k,quantemp27; for(i=1;i=A&*p=Z) /判断字符是否为26字母k=*p-64; /看是26个字符中的哪个字符quantempk+; /字符权值加1 j=0; for(i=1,j=0;i27;i+)if(quant

8、empi!=0)j+; /用于统计字符种类个数strj=i+64; /str按字母表顺序存储出现过的字符quanj=quantempi; return j;void select(HafumanTree HT,int k,int *s1,int*s2) /选择权值最小的两个int i,j; int min1=9999; /声明一个int类型的数值min1,赋个较大的输给它 for(i=1;i=k;i+) /选择权值最小的一个节点(且该节点无父节点) if(HTi.weightmin1)&(HTi.parent=0) j=i;min1=HTi.weight; *s1=j; min1=9999;

9、for(i=1;i=k;i+) if(HTi.weightmin1)&(HTi.parent=0)&(i!=*s1) /选择权值最小的一个节点(且该节点无父节点) j=i;min1=HTi.weight; *s2=j;void gjhafumantree(HafumanTree HT,HafumanCode HC,int quan,char str) /构建哈夫曼树int i,s1,s2; for(i=1;i2*num-1;i+) /将所有的节点赋空 HTi.lchild=0;HTi.rchild=0; HTi.parent=0;HTi.weight=0; for(i=1;i=num;i+)

10、/将num个字符的权值赋给num叶节点 HTi.weight=quani; for(i=1;i=num;i+) /将num个字符赋给codenode HCi.ch=stri; i=1; while(i=num) printf(=%c=%d=n,HCi.ch,quani);i+; /输出每个字符的及其权值 for(i=num+1;i=2*num-1;i+) select(HT,i-1,&s1,&s2); /选择两个权值最小的叶节点HTs1.parent=i;HTs2.parent=i; /两个节点指向同一父节点 HTi.lchild=s1;HTi.rchild=s2;HTi.weight=HTs

11、1.weight+HTs2.weight; /父节点的权值为子节点相加(父节点继续放入选择区) void Hafumanencode(HafumanTree HT,HafumanCode HC) int c,p,i; char cdn; /临时数组用于记录字符在哈夫曼树的位置 int start; cdnum=0; /给cd赋个结束符 for(i=1;i0) /根据节点是其父节点的左右子来记录它的位置 cd-start=(HTp.lchild=c)?0:1; c=p; /将父节点转为子节点 strcpy(HCi.bits,&cdstart); /将得到的0、1字串存入结构体HC printf(

12、%c:%sn,HCi.ch,HCi.bits); HCi.len=num-start; /求每个字符0、1编码长度 void coding(HafumanCode HC,char *str) /根据哈夫曼树确定每个字符的0、1代码codeint i,j;FILE *fp; /声明一个文件夹指针fp=fopen(codefile.txt,w); /打开文件夹codefileprintf(密文为:n); while(*str) /字符串未结束时for(i=1;i=num;i+) if(HCi.ch=*str) /判断字符是否在Codenode中存在 for(j=0;jHCi.len;j+) /将c

13、odenode中该字符的1、0代码存入文件夹 fputc(HCi.bitsj,fp); printf(%s,HCi.bits); break; str+; /字符后移printf(n);fclose(fp);char *decode(HafumanCode HC)FILE *fp;char tempstr9999;char *p;static char cdn+1; /char型数组用于存放从文件夹中读取的1、0代码int i,j,k=0,jsjs;fp=fopen(codefile.txt,r);while(!feof(fp) /当文件夹读取没有结束jsjs=0; /判读一个字符是否译码结束

14、for(i=0;(inum)&(jsjs=0)&(!feof(fp);i+) /当一个字符未译完并且文件未读取结束 cdi= ;cdi+1=0; /让cd赋成空格 cdi=fgetc(fp); /读取文件夹中的一个字符 for(j=1;j=num;j+) if(strcmp(HCj.bits,cd)=0) /看cd的字符串是否等于Codenode中的某个密文 tempstrk=HCj.ch;jsjs=1;printf(=%s=,HCj.bits);k+;break; /将译出的字符赋给临时字符串tempstr,标记一个字符译码结束jsjs赋1,跳出循环 tempstrk=0; /赋给临时字符串

15、一个结束标志p=tempstr; /char型指针指向临时字符串 return p;四、运行结果图5 哈夫曼编码与译码运行结果图五、遇到的问题及解决这部分我主要遇到了如下两个问题,其内容与解决方法如下所列: 当我写完程序,输入一段字符串让程序进行译码和解码是出现了一个问题,译出的结果不是想要的结果,但结果又有一定规律,就是结果中的字符都在明文中出现过,可是字符数量明显比明文中的要少很多。首先我认为为题可能出现在我将明文加密后的存储阶段,于是我在程序将0、1代码存入codeflie.txt文件前,先将临时存储字符串数组内容输出,结果发现没有问题,于是我又在,译码阶段将从codeflie.txt读

16、出的内容逐字输出发现结果和存入的一样。输入和输出都一样。问题不是出现在存储上,于是我想到问题可能出现在,我将0、1编码转成明文的时。于是我认真的看了看我的decode函数,添加了些辅助输出,发现有写编码不是原先的一个字符的译文,有的是两个字符的译文译成了一个字符。于是我又输进一段字符将暗文带入函数计算,发现问题出在但一出一个字符后,我的一个循环并没有终止,而是继续向cd数组中添加,1和0;直至cd的长度超过num时才结束,而下次调用该循环式,前面会有几个1、0没有被编译。所以导致了以上的错误,于是我在函数中声明了一个变量jsjs(计算结束),用它在译完一个字符是让函数跳出循环。于是问题得到了解

17、决。 第二个问题再之前就出现了只不过他和前一个问题没有多大的联系,就是在输出明文结果是字符串的结尾变成了: 烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫。一开始我认为他和第一个问题是一个问题,是存储或者译码出错了,但是但第一个问题解决后他依然存在。于是我才确定这是另外的一个问题。根据以晚的经验出现这种情况的原因就几种,存储出错或者输出形式不对,或者输出的位置不对。而第二种情况可以排除,于是我前后看了看程序中的数组的存储和输出。发现存储形式没有问题,于是我想到了第一个问题解决后输出的结果是在正确解码后面加一串乱码。会不会是输出了一段没有赋值区域内的内容,于是我看了看存储输出内容的临时数组发现我声明时声明的

18、长度为300,而一般用不了那么多,而我输出是将str按照字符串形式输出的也就是说,输出结果的前一段时我要的内容而后一段是系统原先的东西,但是也跟着输出了,于是我在str中添加了0结束标志。这样就不会输出我不要的部分了。后来但我输入更长的一段字符串是后没会出现一个字的汉字,于是我又将str的长度加长,是它尽可能存储跟唱的字符串。六、心得体会 通过对上面程序的编写让我积累了一些编程的经验,比如在你编写代码是首先你应在大脑内先构思好你的程序,并在纸上写下你的思路很实现步骤,而不是急于动手编写。有些事没想好就急于容易出现一些逻辑上的错误,最后实现不了又必须重来。其次是数据类型的处理,不能将字符型数据存其他类型的数据,计算机是不识别的,写程序是要严格规范自己对不同类型数据存储和使用,还有就是当一个数据不再被使用时或下次你要用它来存放其他数值时要记得提前赋空,以免数据出错。还有就是在程序中如果你要重复某些步骤时你可以考虑在主程序外建立一个函数来实现这些步骤,这样在主程序中通过调用这个函数来实现这些步骤,这样可以避免编写同样的代码,并简化你的程序代码。除此之外就是大括号的一定要对其,不然一旦代码长度变长,会出现一堆的括号,你可能看的眼花缭乱不知道哪个与哪个是一对,最后影响你的代码检查。

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

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