Huffman编码与解码实现文件压缩与解压.docx

上传人:b****6 文档编号:7441102 上传时间:2023-05-11 格式:DOCX 页数:9 大小:86.99KB
下载 相关 举报
Huffman编码与解码实现文件压缩与解压.docx_第1页
第1页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第2页
第2页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第3页
第3页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第4页
第4页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第5页
第5页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第6页
第6页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第7页
第7页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第8页
第8页 / 共9页
Huffman编码与解码实现文件压缩与解压.docx_第9页
第9页 / 共9页
亲,该文档总共9页,全部预览完了,如果喜欢就下载吧!
下载资源
资源描述

Huffman编码与解码实现文件压缩与解压.docx

《Huffman编码与解码实现文件压缩与解压.docx》由会员分享,可在线阅读,更多相关《Huffman编码与解码实现文件压缩与解压.docx(9页珍藏版)》请在冰点文库上搜索。

Huffman编码与解码实现文件压缩与解压.docx

Huffman编码与解码实现文件压缩与解压

数据结构课程设计报告

 

Huffman编码与解码实现文件压缩与解压

 

学号:

06080711

姓名:

李爱武

指导教师:

陈波

二○一零年九月三日

目录

目录..………………………………….2

目标任务和问题分析..………………………………….3

算法及思想描述..………………………………….3

---------建立哈夫曼树、压缩、解压缩

程序结构及测试流程..………………………………….5

测试结果分析..………………………………….12

收获与体会..………………………………….12

参考文献..………………………………….13

Huffman编码与解码实现文件压缩与解压

一、目标任务与问题分析

1.1目标任务

采用哈夫曼编码思想实现文件的压缩和解压缩功能,可以将任意文件压缩成任意的压缩文件类型,并且能将压缩后的文件解压成相应的源文件。

1.2问题分析

本问题首先应该是利用哈夫曼思想,对需要压缩的文件中的个字符进行频率统计,为了能对任意的文件进行处理,应该所有的文件以二进制的方式进行处理,即对文件(不管包含的是字母还是汉字)采取一个个的字节(unsignedchar)处理,然后根据统计的频率结果构造哈夫曼树,然后对每个字符进行哈夫曼编码,然后逐一对被压缩的文件的每个字符构建的新的哈夫曼编码存入新的文件中即得到的压缩文件。

解压过程则利用相应的哈夫曼树及压缩文件中的二进制码将编码序列译码,对文件进行解压,得到解压文件。

二、算法思想描述

2.1构造哈夫曼树

要利用哈夫曼编码对文本文件进行压缩,首先必须知道期字符相应的哈夫曼编码。

为了得到文件中字符的频率,一般的做法是扫描整个文本进行统计,编写程序统计文件中各个字符出现的频率。

由于一个字符的范围在[0-255]之间,即共256个状态,所以可以直接用256个哈夫曼树节点即数组(后面有节点的定义)空间来存储整个文件的信息,节点中包括对应字符信息,其中包括频率。

2.1.1哈夫曼树节点的设计

用结构体NODE来存储节点的信息,其中有成员频率weight、父节点parent、左枝节点lchild、右枝节点rchild、对应的编码code。

2.1.2文件字符频率处理

由于所有的文件都采用字节存取,字符的最多个数就只能是256个,即ASCII码在0-255之间,而我们用的是数组来存储文件的信息,我们用NODE[256]来统计文件的所有信息。

此操作的过程中应用的关键技术在于我们直接用该数组

的下标来保存每个字符的信息,原因在于字符为单字节,正好与每个下标0-255对应。

这样既减少了内存的开销,而且还精简了代码。

我们只需要在每读一个字节(inByte)的同时,让对应的NODE[inByte]加一就可以达到统计字符频率的效果。

2.1.3构建哈夫曼树

哈夫曼树的存储结构采用双亲孩子表示法,即用结构体数组来实现。

为了让程序设计更加的精简,我们直接用扩大节点数组的方法来保存,由于原来的256个叶子节点会产生255个父亲节点,所以我们将原来的空间为256的数组扩大为NODE[511]来构建哈弗曼树。

扩充空间后,我们开始初始化哈夫曼树,即通过上述统计的叶子节点循环提取哈弗曼中权值最小的两个节点,将它们合并起来,组成一个新的根节点,直到最后哈夫曼树只剩下一个根节点为止。

2.1.4获得哈夫曼编码

采用字节进行编码,每个字符的Huffman编码是一个0/1串,最高效的方法是采用的是采用位串的形式,我们先定义每个字符的编码为一个字符串。

我们从哈夫曼树的根节点开始循环遍历,并给每个节点设置一个编码标志,约定如果没编码状态为0,,左孩子已编码状态为1,右孩子已编码状态为2,首先如果编码状态为0从左孩子开始,如果有左孩子,那么编码字符串加上“0”字符,状态设为1,如果没有左孩子并且没有右孩子则该节点的编码结束;接着考虑没有左孩子但有右孩子并且右孩子没编码的情况,如果有右孩子则编码字符串加“1”,编码状态为2,如果做有节点都已编码,则再对该节点的兄弟节点进行编码,因为此时除了最后面一个编码字符外,兄弟节点的前部分编码和已完成编码的节点是相同的,没必要重复遍历。

2.1.5将文件进行二进制压缩

以上只能得到文件中每个字符的0/1形式的字符串编码,然而如果想要文件起到真的压缩效果,还必须在读取被压缩文件得到每个字节的编码的同时将字符串编码转化拼接成八位的字节存入压缩文件中。

为此我们采用的是将编码字符串切割成长度为八的字串,然后再将其通过从低位遍历如果为“1”则与字节(inByte)值为00000000取“或“操作再向左将字节移一位,依次循环八次得到相应的二进制字节存入压缩文件,如果最后剩余不足八位,另作处理即将其长度也在最后以二进制输入,达到压缩的效果。

2.1.6将二进制压缩文件解压

与压缩文件相似,我们以字节依次读取压缩文件的每个字节(inByte),一次取字节的每一位,采用的方法是将字节(inByte)与128(10000000)进行“与“操作,得到每位,如果为0,从Huffman树的左孩子下行,反之,从右孩子下行,再将(inByte)向右移一位,不断循环,当循环到孩子节点的下标小于256时,将下标以二进制的方式输入到解压文件中即可。

如果到最后一个字节,其循环次数由其长度决定,于是起到减压的效果。

三、程序结构及测试流程

3.1类的设计

3.1.1HuffmanTree类

此类用以实现各种算法,其设计如下:

 

此类为实现压缩与解压缩操作的各种方法,首先通过成员GetWeight函数计算出需要压缩的文件字符及对应的频率,然后通过BuildTree成员函数根据文件的字符及对应频率建立哈夫曼树,然后调用GenerateCode得到每个字节的编码,最后通过Compress函数进行压缩文件,最终利用可利用Decompress函数进行解压测试,其中SelectMiin为辅助函数,选取建立Huffman树之前权重最小的两个节点。

3.1.2Application类

此类产生的对象用来执行用户选择的各种操作,其设计如下:

此类的公有方法Run启动整个应用程序,供用户选择各种操作,达到了人机交互的效果。

3.1.3核心函数的说明

压缩函数:

voidHuffmanTree:

:

Compress(constchar*fn1,constchar*fn2)//压缩

{

ifstreaminfile(fn1,ios:

:

binary);//以二进制方式打开文件

ofstreamoutfile(fn2,ios:

:

binary);

infile.unsetf(ios:

:

skipws);//设置文件得读取空白符

stringwaiting="";//编码字符串

unsignedcharinByte,outByte;//用来构建读入输出字节

intI;

while(infile.good())//判断文件没结束,不存在或没有打开失败等

{

while(waiting.length()<8)

{

if(!

infile.good())

break;

infile.read((char*)&inByte,sizeof(unsignedchar));//读取一个字节

if(infile.good())

waiting+=tree[inByte].code;//得到该字节的huffman编码

}

while(waiting.length()>7)

{

outByte='\0';//字节八位清零

for(i=0;i<8;i++)

{

outByte=outByte<<1;//左移一位。

原因:

字符串编码的首位为二进制的最高位

if(waiting[i]=='1')

outByte|=1;//当编码为"1",通过或操作符将其添加到字节的最低位

}

outfile.write((char*)&outByte,sizeof(unsignedchar));//将得到的字节存入文件

waiting=waiting.substr(8);//编码字符串去除已存储的前八位构成新的字符串

}

if(infile.eof())

{

if(waiting.length()==0)

outByte=static_cast(0);

else//文件结束,但还剩最后一段不满八位的编码

{

outByte='\0';

for(i=0;i<(int)waiting.length();i++)

{

outByte=outByte<<1;

if(waiting[i]=='1')

outByte|=1;

}

outByte=outByte<<(8-waiting.length());//将编码字段从尾部移到字节的高位

outfile.write((char*)&outByte,sizeof(unsignedchar));//存入最后一个字节

outByte=static_cast(waiting.length());

}

outfile.write((char*)&outByte,sizeof(unsignedchar));//存入最后编码的长度

}

}

infile.setf(ios:

:

skipws);

infile.close();//关闭文件

outfile.close();

}

解压缩函数:

voidHuffmanTree:

:

Decompress(constchar*fn1,constchar*fn2)

{

ifstreamifile(fn1,ios:

:

binary);

ofstreamofile(fn2,ios:

:

binary);

ifile.unsetf(ios:

:

skipws);

ifile.seekg(0,ios:

:

beg);//将文件指针置于文件开始

inti;

//for(i=0;i<256;i++)

//ifile.read((char*)&(tree[i].weight),sizeof(long));

//this->BuildTree();

unsignedcharinbyte1=0,inbyte2=0,inbyte3=0;

intp=510,len=8;

ifile.read((char*)&inbyte1,sizeof(inbyte1));//读取第一个字节

ifile.read((char*)&inbyte2,sizeof(inbyte2));//读取第二个字节

if(ifile.eof())//考虑只有两个字节

{

len=static_cast(inbyte2);

for(i=0;i

{

if(inbyte1&128)

p=tree[p].rchild;

else

p=tree[p].lchild;

inbyte1=inbyte1<<1;

if(p<256)//小于256说明是要找的字节

{

p=static_cast(p);

ofile.write((char*)&p,sizeof(unsignedchar));

}

}

}

while(!

ifile.eof())//没到文件结尾

{

ifile.read((char*)&inbyte3,sizeof(inbyte3));//读取下个字节

if(!

ifile.good())//考虑最后一个字节的情况,计算其对应的编码的长度

len=static_cast(inbyte2);

else

len=8;

for(i=0;i

{

if(inbyte1&128)

p=tree[p].rchild;

else

p=tree[p].lchild;

inbyte1=inbyte1<<1;

if(p<256)

{

p=static_cast(p);

ofile.write((char*)&p,sizeof(unsignedchar));

p=510;

}

}

inbyte1=inbyte2;

inbyte2=inbyte3;

}

ifile.setf(ios:

:

skipws);//文件流复位

ifile.close();

ofile.close();

}

3.2执行效果

运行改程序弹出菜单,供用户选择,其效果如下:

选择菜单1提示用户输入被压缩文件名和压缩后的文件名运行结果如下:

选择菜单2进行解压操作得到的界面如下:

需要退出操作,用户可以选择菜单0

四、测试结果

此压缩软件起到了良好的压缩效果,部分测试数据的结果如下表:

表《一》

表《一》

压缩前文件大小

压缩后文件大小

压缩率

a.doc(50KB)

b.hfm(24KB)

105%

t.txt(6KB)

t1.law(5KB)

20%

e.xls(106KB)

E1.hfm(51KB)

110%

从以上结果可以看出此软件对较大的文件效果跟明显,这是与事实相符的,因为文件越大,字符的重复率越高,根据此软件实现算法,频率越高的字符其编码反而短,效果自然更明显。

五、收获与体会

(1)哈夫曼编码有一些缺点:

对于短的文件进行编码意义不大,因为较短的文件字节的重复率不是很高。

(2)哈夫曼进行解压缩软件设计时候,其核心是哈夫曼树,它编码和译码的纽带。

该压缩软件采用的正是哈夫曼算法,建立哈夫曼树,对原文件进行哈夫曼编码,通过将哈夫曼算法应用于压缩软件,更进一步理解哈夫曼树的建立和对各个叶子节点的编码。

哈夫曼技术对多种文件压缩率很高,对于二进制这种法也很成功。

(3)该程序中因为涉及文件操作,多次读写文件,为了保证操作后的文件与原文件的一致性,在打开或者建立新文件时都是以二进制进行操作,着不仅加深了我们对文件操作的理解,也在程序中体现了完整性,并且将解压后的文件和原文件相比较具有一致性,体现的程序的健壮性

(4)通过这次课程设计,强化了我结构化编程的思想,对复杂问题的数据结构化有了更深刻的了解,对数据结构有了更深的理解,提高了我对算法的设计层次。

六、参考文献

(1)数据结构教程(C++版)/吉根林,陈波主编,—北京:

电子工业出版社,2009.2

(2)现代计算机2008年下半月11期

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

当前位置:首页 > 人文社科 > 法律资料

copyright@ 2008-2023 冰点文库 网站版权所有

经营许可证编号:鄂ICP备19020893号-2