哈夫曼编码
本来是想着慢慢找时间按顺序补上这些数据结构的内容,但前几天有朋友找我写一个计算出哈夫曼编码的程序(课程作业吧,好像~哈哈哈),所以就先把哈夫曼树的东西写下来。
先来介绍一下哈夫曼编码吧
哈夫曼树,二叉树的带权路径长度达到最小,称这样的二叉树为最优二叉树,也称为哈夫曼树(Huffman Tree)。哈夫曼树是带权路径长度最短的树,权值较大的结点离根较近。
哈夫曼编码则是根据字符出现频率(权重分布),用最少的位数来表示。(字符出现多的,用较短编码;字符出现少的,用较长编码)。
举个例子:好比这个字符串“abcddddddd”,如果用相同长度的编码来表示,4个不同数字则选用2位(a->00,b->01,c->10,d->11),那也就是需要 10 * 2 = 20长度;
而使用哈夫曼编码,则根据权重(a->010,b->011,c->00,d->1),也就是需要:3+3+2+1*7 = 15长度。
在这个例子中相比平均编码则少了25%(当然这个例子比较夸张啦哈哈哈。权重分布一般不会这样,在这里主要是想说明哈夫曼编码能起到压缩编码长度的作用)
但是在取哈夫曼编码的时候,要避开前缀(有可能某个数的编码刚好是另外某个数的编码前缀),所以可以通过下面的方法计算出来。
一、基于建哈夫曼树的方法
算法过程:
①将输入的数值,每一个数值,分别都建立成一棵树(只有根节点的树),就形成了森林。
②将森林排序( 按权重,从大到小排序,选用插入排序即可,因为之后的排序每一次都只插入一个数值,只需要移动少量即可)
③取出后两棵树n1、n2,新创建一个根节点,然后将n1、n2作为其左右孩子,从而构成一棵新树(由新建根节点+n1(lchild)+n2(rchild)组成),将这棵新树插入森林中。
④重复②③,直到森林中只剩下一棵树,此时该树就是所求哈夫曼树。
⑤遍历每一个叶子(每一个叶子就是输入数值),根据遍历路径(往lchild则 +‘1’,往rchild 则 + ‘0’),每一个叶子对应的数值+编码result,即是其哈夫曼编码。
代码实现:C++
#include <bits/stdc++.h>
using namespace std;
typedef struct tree_node {
int temp; //存数值/对应字符
int weight; //存权重
tree_node* lchild;
tree_node* rchild;
tree_node(int t = -1, int w = 0) { //非叶子结点,temp都等于-1
temp = t;
weight = w;
lchild = NULL;
rchild = NULL;
}
};
class huffma {
public:
huffma() {
tree = new tree_node;
}
void destory_tree(tree_node* ntree) {
if (ntree != NULL) {
tree_node* lc = ntree->lchild;
tree_node* rc = ntree->rchild;
delete ntree;
if (lc != NULL)destory_tree(lc);
if (rc != NULL)destory_tree(rc);
}
}
~huffma() {
destory_tree(tree);
}
void input(int* data, int len) { //一开始是选用vector<int>,但是朋友说输入是需要数组,所以也就在这里转一下vector<int> 哈哈哈
vector<int>data_in(data, data + len);
make_trees(data_in); //建树
}
void output() { //输出
vector<int> a;
print_result(tree,a); //输出编码
}
private:
tree_node* tree; //做根结点
vector<tree_node*> trees; //森林
void sort_trees() { //森林排序:插入排序
for (int i = 1; i < trees.size(); i++) {
tree_node* change = trees[i];
int j = i - 1;
while (change->weight > trees[j]->weight) {
trees[j + 1] = trees[j];
j--;
if (j == -1)break;
}
trees[j + 1] = change;
}
}
void built_it() { //建树
int len = trees.size();
sort_trees();
while (len >= 3) {
tree_node* n1 = trees.back();
trees.pop_back();
tree_node* n2 = trees.back();
trees.pop_back();
tree_node* new_node = new tree_node(-1, n1->weight + n2->weight);//temp = -1,表示非叶子结点
new_node->lchild = n1;
new_node->rchild = n2;
trees.push_back(new_node);
sort_trees();
len = trees.size();
} //剩下最后两棵,直接作为创建的根节点的左右孩子
tree_node* n1 = trees.back();
trees.pop_back();
tree_node* n2 = trees.back();
trees.pop_back();
tree->lchild = n1;
tree->rchild = n2;
}
void print_result(tree_node* p, vector<int> a) { //递归,找出编码
if (p->temp != -1) {
cout << p->temp << "->";
for (int i = 0; i < a.size(); i++) cout << a[i];
cout << endl;
}
a.push_back(0);
if (p->lchild != NULL)print_result(p->lchild, a);
a.pop_back();
a.push_back(1);
if (p->rchild != NULL)print_result(p->rchild, a);
a.pop_back();
}
void make_trees(vector<int> data) { //vector<int>存入森林,并built_it()
for (int i = 0; i < data.size(); i++) {
tree_node* new_tree = new tree_node(data[i], data[i]);
trees.push_back(new_tree);
}
built_it();
}
};
int main() {
huffma s;
int data[8] = { 2,5,6,8,13,19,25,36 }; //输入:元素表示:字符和权重(相同)
int len = sizeof(data) / sizeof(data[0]);
s.input(data, len);
s.output();
return 0;
}
二、不通过正常建树的方法
算法过程:
分别用两个数组in、out,来存下结点。
用in数组存下输入值,模拟方法一中的森林。
用out数组存下每次从in中取出来的结点。
过程:
①将输入数组存入 数组in
②对 数组in 进行排序(从大到小,这样在后两位、插入时都不需要移太多位)
③取出 数组in 的后两位,分别对其.result 赋值0或1,存到 数组out中,新建一个元素(=两元素的权值合,并记录子元素的位置,也即由两位合成后,记录这两位在 数组out 中的位置),将新建的元素插入 数组in。
④重复②③,知道 数组in 为空。
⑤此时由后往前遍历 数组out,对每一位元素进行操作:根据该元素的记录位置,找到 数组out 中对应的元素,添加上该元素的赋值。(按照下图来说明比较容易)
/*
好吧,其实两个方法是同样的思路,但是前一种会去体系的建成一棵哈夫曼树后遍历找到哈夫曼编码,而后一种方法则是通过建立各元素之间的关系来找到哈夫曼编码。
当然 ,第一个方法涉及各种指针,其中还没有去delete掉那部分不使用到的内存,造成内存泄漏,所以如果数量级大了可能就内存溢出了,所以还有待修改优化。
至于第二种方法,其实就是运用了方法一的计算方法,但是没有建成一棵哈夫曼树,所以如果在其他有关树的操作时则会很麻烦。所以只是来算出这个哈夫曼编码而已哈哈哈。
当然方法一比较直接,编程方法也比较简单。相比方法二则需要理清晰其间关系与操作顺序。在此建议:将整个过程想法写出,再编程。这是个好习惯!!!可以省去很多麻烦,特别是逻辑上的错误
*/
代码实现:C++
#include <bits/stdc++.h>
using namespace std;
typedef struct tree_node {
int temp; //存元素:字符
int weight; //权重
vector<int> result; //存其result
vector<int> flag; //存其两个合成的子元素的out数组位置
tree_node(int temp = -1, int weight = 0) { //新建元素皆temp = -1
this->temp = temp;
this->weight = weight;
}
};
void node_sort(vector<tree_node>& data_in) { //插入排序
for (int i = 1; i < data_in.size(); i++) {
tree_node change = data_in[i];
int j = i - 1;
while (change.weight > data_in[j].weight) {
data_in[j + 1] = data_in[j];
j--;
if (j == -1)break;
}
data_in[j + 1] = change;
}
}
void get_result(vector<tree_node>& data_out,int ing) { //根据flag,去写对应out[]的result
if(data_out[ing].flag.size() > 0){
for(int i = 0;i< data_out[ing].result.size();i++){
data_out[data_out[ing].flag[0]].result.push_back(data_out[ing].result[i]);
data_out[data_out[ing].flag[1]].result.push_back(data_out[ing].result[i]);
}
}
}
void opperate(int* data, int len) { //整个过程
vector<tree_node>data_in; //数组in
vector<tree_node>data_out; //数组out
for (int i = 0; i < len; i++) { //读入输入
tree_node node = tree_node(data[i], data[i]);
data_in.push_back(node);
}
node_sort(data_in); //排序
int k = 0;
while (data_in.size() > 1) { //重复②,③
tree_node n1 = data_in.back();
data_in.pop_back();
n1.result.push_back(0); //较小元素,则添加result '0'
data_out.push_back(n1);
tree_node n2 = data_in.back();
data_in.pop_back();
n2.result.push_back(1); //较大元素,则添加result '1'
data_out.push_back(n2);
tree_node n3 = tree_node(-1, n1.weight + n2.weight);
n3.flag.push_back(k++);
n3.flag.push_back(k++);
data_in.push_back(n3);
node_sort(data_in);
}
//从后往前遍历
for (int i = data_out.size()-1; i >= 0; i--)get_result(data_out, i);
//依次输出out数组里(temp !=1 的元素)
for (int i = 0; i < data_out.size(); i++) {
if (data_out[i].temp != -1) {
cout << data_out[i].temp << "->";
for (int j = data_out[i].result.size() - 1; j >= 0; j--)cout << data_out[i].result[j];
cout << endl;
}
}
}
int main() {
int data[8] = { 2,5,6,8,13,19,25,36 }; //输入:元素表示:字符和权重(相同)
int len = sizeof(data) / sizeof(data[0]);
opperate(data, len);
return 0;
}
两个程序运行结果分别如下:
(左为方法一,右为方法二。结果一致,但方法二的结果是从权重低的开始排起,所以在使用中,“对比”、“查找”等操作则会方便很多,从时间上来说,两个方法差别不大,测试了128个字符时,并没有多大的时间差别)
可优化部分:
①选用插入排序,主要是考虑到后续插入时只插入一个,都是O(n),应该时间消耗不是很大。可能选用其他排序更优,但是有一点值得提的是,插入排序是稳定排序(稳定排序,当新建结点 与 输入元素 权重相同时,可以优先取新建结点,这样会使较深层叶子结点用少一丢丢的编码长度,当然我只是脑补一下这个结果,真实情况再来探讨,所以先保留为待优化空间)
想来最终还是选择插入排序,因为在有序序列的单次插入时,最快的还是插入排序。
②在修改下一层的result时,总是用 目前已经修改好(2020.12.23修改),修改主要是用同一个vector在每次左孩子递归前push_back(0),递归完左孩子再pop_back()掉,再push_back(1),这样改的结果就是很好的省去了一直重复复制vector的开销,但是不好的地方就是只能在过程中找出结果,并不能直接保存到每一个结点里。
for(int i = 0;i< data_in.size();i++)data.push_back( data_in[i]),这里重复度很高。这里有很大的时间浪费在这里。每下一个结点就得重新复制一遍,所以这里有很大的优化空间。
同时,此次修改还修改了关于析构函数方面,因为之前只想着树建完输出就整个程序结束了就全部释放了,所以没去考虑内存泄漏的事。现在补上析构函数~哈哈哈哈
目前就这样吧。哈哈,有其他待优化空间或者其他好的优化方法或者其他更好的解决方法,都很愉快可以来分享~