作者:jostree 转载请注明出处 http://www.cnblogs.com/jostree/p/4096079.html

使用优先队列实现,需要注意以下几点:

1.在使用priority_queue时,内部需要存储哈夫曼树节点的指针,而不能是节点。因为构建哈夫曼树时,需要把其左右指针指向孩子,而如果储存的是节点,那么孩子的地址是会改变的。同理节点应当使用new在内存中开辟,而不能使用vector,原因是vector在数组大小为2整数次幂时,大小会倍增,开辟新数组并把老数组的数字copy过去,从而也会导致地址变化。

2.优先队列对指针的排列,需要额外写一个比较函数来比较指针指向的节点的大小。bool operator () (wcnode * node1, wcnode * node2) return node1->lessthan(node2);并在定义优先队列时使用这种方法:    priority_queue <wcnode*, vector<wcnode*>, compare> 第一个参数是节点类型,第二个参数是优先队列的储存结构,第三个参数是比较函数。

3.C++在写入文件时,由于只能按字节写入,因此需要把8个bit位转化为一个字节,最后不足8位用0补齐,并记录文件总bit数,便于解码。然后写入文件。另写入二进制文件可以使用ofstream out("output.txt",std::ofstream::binary);

4.哈夫曼编码信息包括每种字符的映射,和该文件的总bit数。

其代码如下:

 #include <cstdio>
#include <cstdlib>
#include <iostream>
#include <cstring>
#include <fstream>
#include <queue>
#include <map>
#include <vector>
using namespace std;
class compare; class wcnode
{
public:
friend class compare;
char word;
int count;
wcnode* left;
wcnode* right;
bool lessthan (const wcnode *w)const
{
return count > w->count;
}
wcnode(char w='\0', int c=, wcnode* l=NULL, wcnode * r=NULL)
{
word = w; count = c; left = l; right = r;
}
}; class compare
{
public:
bool operator () (wcnode * node1, wcnode * node2)
{
return node1->lessthan(node2);
}
}; void preorder(wcnode *head, vector<bool> rec, map<char, vector<bool> > & res)
{
if( head->left == NULL && head->right == NULL )
{
res[head->word] = rec;
return;
}
vector<bool> l = rec;
l.push_back();
vector<bool> r = rec;
r.push_back();
if(head->left != NULL) preorder(head->left, l, res);
if(head->right != NULL) preorder(head->right, r, res);
}
map<char, vector<bool> > encode(map<char, int> &wordcount)
{
map<char, vector<bool> > res;
priority_queue <wcnode*, vector<wcnode*>, compare> pq;
map<char, int>::iterator t;
wcnode *tmp;
wcnode *t1, *t2, *t3; for( t = wordcount.begin() ; t != wordcount.end() ; t++ )
{
tmp = new wcnode();
tmp->word = t->first;
tmp->count = t->second;
pq.push(tmp);
}
while( pq.size() > )
{
t1 = pq.top();
pq.pop();
t2 = pq.top();
pq.pop();
t3 = new wcnode();
t3->count = t1->count + t2->count;
t3->left = t1;
t3->right = t2;
pq.push(t3);
}
wcnode *huffmanhead = pq.top();
vector<bool> rec;
preorder(huffmanhead, rec, res);
map<char, vector<bool> >::iterator it;
for( it = res.begin() ; it != res.end() ; it++ )
{
cout<<it->first<<":";
for( int i = ; i < it->second.size() ; i++ )
{
cout<<it->second[i];
}
cout<<", ";
}
return res;
} void output(string s, string passage, map<char, vector<bool> > res)
{
ofstream out(s.c_str());
vector<bool> bit;
for( int i = ; i < passage.size() ; i++ )
{
vector<bool> tmp = res[passage[i]];
for( int i = ; i < tmp.size(); i++ )
{
bit.push_back(tmp[i]);
}
}
char outputchar = ;
for( int i = ; i < bit.size() ; i++ )
{
if( i % == )
{
out.write(&outputchar, sizeof(outputchar));
outputchar = ;
}
outputchar = outputchar + bit[i];
outputchar = outputchar * ;
}
if( outputchar != )
{
out.write(&outputchar, sizeof(outputchar));
}
out.close();
}
int main(int argc, char *argv[])
{
char tmp;
ifstream in("Aesop_Fables.txt");
map <char, int> wordcount;
map <char, vector<bool> > res;
string passage;
while( in.get(tmp) )
{
passage += tmp;
if( wordcount.count(tmp) == )
{
wordcount[tmp] = ;
}
else
{
wordcount[tmp]++;
}
}
res = encode(wordcount);
output("outAesop.txt", passage, res);
in.close();
}

 

Huffman Coding 哈夫曼编码的更多相关文章

  1. Huffuman Coding (哈夫曼编码)

    哈夫曼编码(Huffman Coding),又称霍夫曼编码,是一种编码方式,哈夫曼编码是可变字长编码(VLC)的一种.Huffman于1952年提出一种编码方法,该方法完全依据字符出现概率来构造异字头 ...

  2. 哈夫曼(Huffman)树+哈夫曼编码

    前天acm实验课,老师教了几种排序,抓的一套题上有一个哈夫曼树的题,正好之前离散数学也讲过哈夫曼树,这里我就结合课本,整理一篇关于哈夫曼树的博客. 主要摘自https://www.cnblogs.co ...

  3. 霍夫曼编码(Huffman Coding)

    霍夫曼编码(Huffman Coding)是一种编码方法,霍夫曼编码是可变字长编码(VLC)的一种. 霍夫曼编码使用变长编码表对源符号(如文件中的一个字母)进行编码,其中变长编码表是通过一种评估来源符 ...

  4. 哈夫曼编码(Huffman coding)的那些事,(编码技术介绍和程序实现)

    前言 哈夫曼编码(Huffman coding)是一种可变长的前缀码.哈夫曼编码使用的算法是David A. Huffman还是在MIT的学生时提出的,并且在1952年发表了名为<A Metho ...

  5. 哈夫曼编码的理解(Huffman Coding)

    哈夫曼编码(Huffman Coding),又称霍夫曼编码,是一种编码方式,可变字长编码(VLC)的一种.Huffman于1952年提出一种编码方法,该方法完全依据字符出现概率来构造异字头的平均长度最 ...

  6. 赫夫曼\哈夫曼\霍夫曼编码 (Huffman Tree)

    哈夫曼树 给定n个权值作为n的叶子结点,构造一棵二叉树,若带权路径长度达到最小,称这样的二叉树为最优二叉树,也称为哈夫曼树(Huffman Tree).哈夫曼树是带权路径长度最短的树,权值较大的结点离 ...

  7. 哈夫曼树(Huffman Tree)与哈夫曼编码

    哈夫曼树(Huffman Tree)与哈夫曼编码(Huffman coding)

  8. 哈夫曼(huffman)树和哈夫曼编码

    哈夫曼树 哈夫曼树也叫最优二叉树(哈夫曼树) 问题:什么是哈夫曼树? 例:将学生的百分制成绩转换为五分制成绩:≥90 分: A,80-89分: B,70-79分: C,60-69分: D,<60 ...

  9. (转载)哈夫曼编码(Huffman)

    转载自:click here 1.哈夫曼编码的起源: 哈夫曼编码是 1952 年由 David A. Huffman 提出的一种无损数据压缩的编码算法.哈夫曼编码先统计出每种字母在字符串里出现的频率, ...

随机推荐

  1. 笔记本禁用自带键盘攻略-------针对shift默认按下的解决方案

    长期以来楼主一直被一个问题困扰,就是win8进入界面,输密码时开大写其实是小写,开小写是大写.进入系统以后shift键是默认按下的.一直以为是电脑中毒了.上网查了一些东西,发现可能是因为键盘硬件方面的 ...

  2. HOG特征-理解篇

    网上去找关于HOG的资料,发现理解性的较少,并且较为冗长,为方便大家理解便自己写了篇,希望能对奋斗在特征提取第一线的同志们有所帮助: HOG即histogram of oriented gradien ...

  3. android的ListView做表格添加圆角边框

    边框,圆角,都可以实现的 在drawable目录下添加view_yuan_morelist.xml,设置控件的边框代码.如下: <?xml version="1.0" enc ...

  4. 【转贴】gdb中的信号(signal)相关调试技巧

    一篇不错的帖子,讲的是gdb中的信号(signal)相关调试技巧 转自Magic C++论坛  http://www.magicunix.com/index_ch.html  http://www.m ...

  5. jQuery 学习笔记(未完待续)

    一.jQuery概述    宗旨: Write Less, Do More.    基础知识:        1.符号$代替document.getElementById()函数        2.使 ...

  6. careercup-C和C++ 13.7

    13.7 写一个函数,其中一个参数是指向Node结构的指针,返回传入数据结构的一份完全拷贝. Node结构包含两个指针,指向另外两个Node. C++实现代码: typedef map<Node ...

  7. centos6.5 安装jdk7和tomcat7

    JDK安装: 安装说明 系统环境:centos-6.5安装方式:rpm安装软件:jdk-7-linux-x64.rpm下载地址:http://www.oracle.com/technetwork/ja ...

  8. swift 如何使用OC中宏的功能

    swift中没有宏的概念,那么我们在swift使用宏的功能来提高效率呢? 一.使用关键字 let 来声明一个常量 存储相应的值,以下代码声明了常量  myColor 来存储一种指定的颜色 let my ...

  9. SVN理解

    先说说什么是branch.按照Subversion的说法,一个branch是某个development line(通常是主线也即trunk)的一个拷贝,见下图: branch存在的意义在于,在不干扰t ...

  10. swift项目中嵌入oc

    参考资料 需要注意的是  与oc包含swift不同的是  swift包含oc需要在桥接文件中包含要使用的oc的头文件 demo:swiftPlayOc(提取码:37c6)