压缩率

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

一、 问题分析和任务定义 1.1设计任务 采用哈夫曼编码思想实现文件的压缩和恢复功能,并提供压缩前后的占用空间之比。要求 (1)描述压缩基本符号的选择方法。 (2)运行时的压缩原文件的规模应不小于5K。 (3)提供恢复文件与原文件的相同性对比功能。 1.2问题分析 本课题是利用哈夫曼编码思想,设计对一个文本文件(.txt)中的字符进行哈夫曼编码,生成编码压缩文件,并且还可将一个压缩后的文件进行解码还原为原始文本文件(.txt)。 在了解哈夫曼压缩解压缩原理之前,首先让我们来认识哈夫曼树。哈夫曼树又称最优二叉树,是带权路径长度最小的二叉树。 在文本文件中多采用二进制编码。为了使文件尽可能的缩短,可以对文件中每个字符出现的次数进行统计。设法让出现次数多的字符二进制码短些,而让那些很少出现的字符二进制码长一些。若对字符集进行不等长编码,则要求字符集中任一字符的编码都不是其它字符编码的前缀。为了确保哈夫曼编码的唯一性,我们可以对它的左右子树的大小给予比较限定,如:左子树的权值小于右子树的权值。哈夫曼树中的左右分支各代表‘0’和‘1’,则从根节点到叶子节点所经历的路径分支的‘0’和‘1’组成的字符串,为该节点对应字符的哈夫曼编码。 统计字符中每个字符在文件中出现的平均概率(概率越大,要求编码越短)。利用哈夫曼树的特点:权越大的叶子离根越近,将每个字符的概率值作为权值,构造哈夫曼树。则概率越大的节点,路径越短。哈夫曼译码是从二进制序列的头部开始,顺序匹配成共的部分替换成相应的字符,直至二进制转换为字符序列。 哈夫曼用于文件解压缩的基础是在压缩二进制代码的同时还必须存储相应的编码,这样就可以根据存储的哈夫曼编码对压缩代码进行压缩。总之,该课题的任务应该是首先要打开要压缩的文本文件并读出其字符出现的频率,以其为权值构建哈夫曼树。其次要找到构建压缩功能的方法,在构建哈夫曼树的基础上进行编码,改变字符原先的存储结构,以达到压缩文件的目的,以外还有存储相应的哈夫曼编码,为解压缩做准备。 1.3测试用数据 本实验的数据是通过读入一个名为huffman.txt的文本文档,文档中内容为字符型数据



二、 概要设计和数据结构的选择 以下是在任务分析对题意的理解做出的概要设计和对数据结构的选择: 1、 数据结构定义 //huffman树的结点结构体 typedef struct HTnode { long weight; //记录结点的权值 int parent; //记录结点的双亲结点位置 int lchild; /结点的左孩子 int rchild; //结点的右孩子 int

*code; //记录该结点的huffman编码 int codelen; //记录该结点huffman编码的长度 //初始化结点,令其权值为无穷大,无双亲及左右孩子 HTnode() { weight = MAX; parent = -1; lchild = -1; rchild = -1; codelen = 0; } }HTnode; 2、 定义huffman数类及其函数 class huffmanTree { public: huffmanTree(); virtual ~huffmanTree(); bool count(char *input); //压缩时统计各字符出现的次数,将其写入对应结点的权值 void create(); //压缩时根据各结点的权值构造huffman树
void code(); //压缩时利用huffman树计算每个字符的huffman编码 void printcode(); //列出每个字符的huffman编码 void addbit(int bit); //压缩时对一个未满8个bit的byte中加入一个bit void resetbyte(); //将byte清空 bool compress(char *input, char *output);//压缩函数,成功返回 true 失败 false bool decompress(char *input, char *output); //恢复函数,成功返回 true 失败false void compare(char *input, char *output); //将原文件与压缩后的文件比较 void compare2(char *input, char *output); //将原文件与恢复后的文件比较 private: int root; //记录根结点的位置 int leafnum; //记录不同字符的个数 HTnode HT[leaf*2-1]; //HTnode结构的数组,用来表示huffman树,树的最大结点个数不会超过leaf*2-1 char byte; //压缩文件时用来缓冲bit的变量 int bitsnum; //byte中bit的个数 int lacknum; //压缩到最后byte中的bit不满8个时填充的0的个数 }; 3、 主程序的流程及模块间关系 主函数实例化huffmanTree类,并实现菜单工具栏,通过用户的选择输入,用switch语句进行分支执行huffmanTree类中功能函数: 1:压缩函数 bool compress(char *input, char *output) 2:恢复函数 bool decompress(char *input, char *output) 3:恢复文件与原文件的对比函数 void compare2(char *input, char *output) 并可在完成相应功能后安全退出,压缩或恢复的文件在同文件夹下生成。 三、 详细设计和编码 核心算法----huffman算法: (1) 根据给定的n个权值{w1,w2,……,wn}构成n棵二叉树的集合F={T1,T2,……,Tn}, 其中每棵二叉树T1中只有一个带权的 w1的根据点,其左右子树均空。 (2) 在F中选取两棵根结点的权值最小的树作为左右子树构造一棵新的二叉 树,且置新的二叉树的根结点的权值为其左右树上根结点的权值之和。 (3) 在F中删除这两棵树,同时将所得到的二叉树加入F中。 (4) 重复(2)(3),直到F中只含一棵树为止。这棵树便是Huffman树。Huffman 树可用于构造代码总长度最短的编码方案。 为了详细说明这个问题,特以下面例子来说明:有四个叶子结点A,B,C,D,分别带权为9,4,5,2,可以构成许多种不同的带权二叉树,但各个带权二叉树的WPL(树的带权路径长度)不同,要想由n个带权叶子

结点所构成的二叉树中,满二叉树或完全二叉树不一定是最优树。权值越大的结点离根越近的二叉树才是最优二叉树(huffman树)。按照上面的算法,则可按照下面图的构造过程生成huffman树。 Huffman树产生流程:
四、 上机调试 以下是我在上机过程中遇到的一些问题及解决方案 开始考虑问题是,要对文件进行压缩,如何才能达到比较好的效果,那就 huffman编码是采用等长编码还是采用不等长问题,采用不登长编码要避免译码的二义性或多义性。假设用0表示字符D,用01表示字符C则当接受到编码串“…01…”,并译到字符0时,是立即译出对应的字符D,还是接着与下一个字符1一起译为对应的字符C,这就产生了二义性。因此,若对某一个字符集进行不等长编码,则要求字符集合中任何一个字符的编码都不能是其他字符编码的前缀。符合此要求的编码叫做前缀编码。显然等长编码是前缀编码,这从等长编码所对应的编码二叉树也可以直接看出,任何一个叶子结点都不可能是其它叶子结点的双亲,也就是说,只有当一个结点是另一个结点的双亲时,该结点的字符编码才会是另一个结点的字符编码的前缀。 为了使不等长编码为前缀编码,可用该字符集中的每个字符作为叶子结点生成一棵编码二叉树,为了获得文件的最短长度,特将每个字符的出现频率作为字符结点的权值赋予该结点上,求出此树的最小带权路径长度就等于文件的最短长度。因此,对文件进行压缩,就可以转化字符集中的所有字符作为叶子结点,字符出现的频率作为权值所产生的 huffman树的问题。 基本思路大致有了后,接下来是对程序的编写工作,程序初步形成后,对其测试,发现了一些语法错误,修正后编译通过。 五、 使用说明 用户进行本程序前,首先要在起工程文件下建立一个待压缩的文本文档,例如:huffman.txt,文档内已有内容,且文档大小大于5K。 运行程序如下图所示


// stdafx.h #include //输入输出头文件 #include //文件操作的类和方法 #include //队列容器 using namespace std; const int leaf = 256; //最多可能出现的不同字符数 const long MAX = 99999999; //表示无穷大 //huffman树的结点结构体 typedef struct HTnode { long weight; //记录结点的权值 int parent; //记录结点的双亲结点位置 int lchild; //结点的左孩子 int rchild; //结点的右孩子 int *code; //记录该结点的huffman编码 int codelen; //记录该结点huffman编码的长度 //初始化结点,令其权值为无穷大,无双亲及左右孩子 HTnode() { weight = MAX; parent = -1; lchild = -1; rchild = -1; codelen = 0; } }HTnode

; //############################################################## //huffmanTree.h //huffman树类 class huffmanTree { public: huffmanTree(); virtual ~huffmanTree(); bool count(char *input); //压缩时统计各字符出现的次数,将其写入对应结点的权值 void create(); //压缩时根据各结点的权值构造huffman树 void code(); //压缩时,利用建好的huffman树计算每个字符的huffman编码 void printcode(); //列出每个字符的huffman编码 void addbit(int bit); //压缩时对一个未满8个bit的byte中加入一个bit
void resetbyte(); //将byte清空 bool compress(char *input, char *output); //压缩函数 成功执行返回 true 失败 false bool decompress(char *input, char *output); //恢复函数 成功执行返回 true 失败 false void compare(char *input, char *output); //将原文件与压缩后的文件比较 void compare2(char *input, char *output); //将原文件与恢复后的文件比较 private: int root; //记录根结点的位置 int leafnum; //记录不同字符的个数 HTnode HT[leaf*2-1]; //HTnode结构的数组,用来表示huffman树,树的最大结点个数不会超过leaf*2-1 char byte; //压缩文件时用来缓冲bit的变量 int bitsnum; //byte中bit的个数 int lacknum; //压缩到最后byte中的bit不满8个时填充的0的个数 }; //############################################################## //huffmanTree.cpp #include "stdafx.h" #include "huffmanTree.h" ////////////////////////////////////////////////////////////////////// // Construction/Destruction ////////////////////////////////////////////////////////////////////// huffmanTree::huffmanTree() { //初始化成员变量 root = 0; leafnum = 0; byte = 0; bitsnum = 0; lacknum = 0; } huffmanTree::~huffmanTree() { for(int i=0; ihuffmanTree::count(char *input)
root = c+384; //读出根结点的位置 for(int i=0; i1) { //还未到最后一个字节 c = q.front(); for(int i=0; i<8; i++) { if(int(c&128)==0) { point = HT[point].lchild; if(HT[point].lchild==-1 && HT[point].rchild==-1) { ofs.put(char(point-128)); point = root; } c = c << 1; } else { point = HT[point].rchild; if(HT[point].lchild==-1 && HT[point].rchild==-1) { ofs.put(char(point-128)); point = root; } c = c << 1; } } q.pop(); }
c = q.

front(); //最后一个字节 for(i=0; i<8-lacknum; i++) { if(int(c&128)==0) { point = HT[point].lchild; if(HT[point].lchild==-1 && HT[point].rchild==-1) { ofs.put(char(point-128)); point = root; } c = c << 1; } else{ point = HT[point].rchild; if(HT[point].lchild==-1 && HT[point].rchild==-1) { ofs.put(char(point-128)); point = root; } c = c << 1; } } q.pop(); ifs.close(); ofs.close(); return true; } //将原文件与压缩后的文件比较 void huffmanTree::compare(char *input, char *output) { ifstream origin, compress; origin.open(input,ios::binary); compress.open(output,ios::binary); if(!origin) { cout << "无法打开文件 " << input << '!' << endl; return; } if(!compress) { cout << "无法打开文件 " << output << '!' << endl; return; } double total1=0, total2=0; char c; while(origin.get(c)) total1++;
while(compress.get(c)) total2++; cout << "原文件大小:" << total1 << " Byte" << endl; cout << "压缩后大小:" << total2 << " Byte" << endl; cout << "压缩率:" << total2/total1*100 << '%' << endl; origin.close(); compress.close(); } //将原文件与恢复后的文件比较 void huffmanTree::compare2(char *input, char *output) { ifstream origin, decompress; origin.open(input,ios::binary); decompress.open(output,ios::binary); double total1=0, total2=0; char c1, c2; bool dif = false; while(origin.get(c1) && decompress.get(c2)) { if(c1!=c2) //依次比较每个字节,不同则将dif标志设为true dif = true; total1++; total2++; } while(origin.get(c1)) { //若原文件还有剩余的数据,将dif设为true dif = true; total1++; } while(decompress.get(c2)) { //若恢复文件还有剩余的数据,将dif设为true dif = true; total2++; } cout << "原文件大小:" << total1 << " Byte" << endl; cout << "恢复文件大小:" << total2 << " Byte" << endl; if(dif==true) cout << "原文件与恢复文件不同!" << endl; else cout << "原文件与恢复文件相同!" << endl; origin.close(); decompress.close(); } //############################################################## //huffman.cpp
#include "stdafx.h" #include "huffmanTree.h" void main() { int choice = 1; char input[255], output[255]; huffmanTree h; while(choice) { cout<<" ***************************************************"<

*"<"; cin >> choice; switch(choice) { case 1: { cout << "请输入待压缩的文件名:"; cin >> input; cout << "请输入压缩后的文件名:"; cin >> output; if( press(input,output)) { h.printcode(); pare(input,output); cout<cin >> input; cout << "请输入恢复后的文件名:"; cin >> output; if (h.decompress(input,output)) cout<> input; cout << "请输入恢复文件的文件名:"; cin >> output; pare2(input,output); } break; case 4: { //执行清屏命令 system("cls"); } break; case 5: break; default: cout << "参数错误!请重新输入" << endl; } cout << endl; } }

相关文档
最新文档