Haffman编码

时间限制:1000 ms  |  内存限制:65535 KB
难度:3
 
描述

哈弗曼编码大家一定很熟悉吧(不熟悉也没关系,自己查去。。。)。现在给你一串字符以及它们所对应的权值,让你构造哈弗曼树,从而确定每个字符的哈弗曼编码。当然,这里有一些小规定:

1.规定哈弗曼树的左子树编码为0,右子树编码为1;

2.若两个字符权值相同,则ASCII码值小的字符为左孩子,大的为右孩子;

3.创建的新节点所代表的字符与它的左孩子的字符相同;

4.所有字符为ASCII码表上32-96之间的字符(即“ ”到“`”之间的字符)。

 
输入
输入包含多组数据(不超过100组) 每组数据第一行一个整数n,表示字符个数。接下来n行,每行有一个字符ch和一个整数weight,表示字符ch所对应的权值,中间用空格隔开。 输入数据保证每组测试数据的字符不会重复。
输出
对于每组测试数据,按照输入顺序输出相应的字符以及它们的哈弗曼编码结果,具体格式见样例。
样例输入
3
a 10
b 5
c 8
4
a 1
b 1
c 1
d 1
样例输出
a:0
b:10
c:11
a:00
b:01
c:10
d:11
题解:让求一颗哈夫曼树,哈夫曼树是先找到最小的两个建一个根节点,这个根节点的权值为两个小的权值和;
加入这个新节点,重新找;用优先队列实现;其根节点即为元素在树中的位置;
代码:
#include<cstdio>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<queue>
using namespace std;
const int INF=0x3f3f3f3f;
#define mem(x,y) memset(x,y,sizeof(x))
#define SI(x) scanf("%d",&x)
#define PI(x) printf("%d",x)
typedef long long LL;
string ans[];
char c[];
struct Node{
char c;
int v;
Node *l,*r;
Node(){
l=r=NULL;
}
Node(char ch,int vv,Node *ll=NULL,Node *rr=NULL){
c=ch;v=vv;l=ll;r=rr;
}
bool operator < (const Node &b) const{
if(v!=b.v)return v>b.v;
else return c>b.c;
}
};
Node e[];
void dfs(Node *now,string s){
if(now->l==NULL&&now->r==NULL){
ans[int(now->c)]=s;return;
}
if(now->l!=NULL)dfs(now->l,s+'');
if(now->r!=NULL)dfs(now->r,s+'');
return ;
}
int main(){
int N,v;
//char s[2];
while(~SI(N)){
if(N==)continue;
for(int i=;i<;i++)e[i].l=e[i].r=NULL;
priority_queue<Node>Q;
getchar();
for(int i=;i<N;i++){
scanf("%c %d",&c[i],&v);getchar();//这里不这样输入就错了
//c[i]=s[0];
Q.push(Node(c[i],v));
}
int tp=;
while(Q.size()>){
e[tp++]=Q.top();
Q.pop();
e[tp++]=Q.top();
Q.pop();
Q.push(Node(e[tp-].c,e[tp-].v+e[tp-].v,&e[tp-],&e[tp-]));
//Node(e[tp-2].c这样插是为了当和等于下一个的v的时候排在前边;
//为什么要再开一个数组e:为了保持队列里面元素的l,r对应关系;
}
Node ss;
ss=Q.top();
//printf("%c %d\n",ss.c,ss.v);
dfs(&ss,"");
//printf("****");
for(int i=;i<N;i++){
//printf("%c:%s\n",c[i],ans[int(c[i])].c_str());
cout<<c[i]<<":"<<ans[int(c[i])]<<endl;
}
}
return ;
}

数据结构课后又写了下:优先队列那点要用仿函数写;

代码:

#include "handsomecui.h"
#include<queue>
#include<vector>
struct Node{
int v;
char c;
Node *l, *r;
Node(){
l = NULL;
r = NULL;
}
void init(int v, char c, Node *l = NULL, Node *r = NULL){
this->v = v;
this->c = c;
this->l = l;
this->r = r;
}
// bool operator < (const Node *a) const{
// return v > a->v;//这里的排序有问题
// }
};
struct cmp{
bool operator() (const Node *a, const Node *b) const{
if(a->v != b->v)
return (a->v) > (b->v);
else
return a->c > b->c;
}
};
priority_queue<Node*, vector<Node*>, cmp>Q;
string ans[];
void dfs(Node *root, string s){
if(root->l == NULL && root->r == NULL){
ans[root->c] = s;
return;
}
if(root->l != NULL){
dfs(root->l, s + '');
}
if(root->r != NULL){
dfs(root->r, s + '');
}
}
char s[];
void print(char c){
PI(c, ":");
PI(ans[c] , "\n");
}
int main(){
int n;
int v;
while(~scanf("%d", &n)){
if(n == )
continue;
getchar();
while(!Q.empty())
Q.pop();
Node *a, *b, *c;
for(int i = ; i < n; i++){
a = new Node();
scanf("%c%d", &s[i], &v);
getchar();
a->init(v, s[i]);
Q.push(a);
}
while(Q.size() > ){
a = Q.top();
Q.pop();
b = Q.top();
Q.pop();
c = new Node();
c->init(a->v + b->v, a->c, a, b);
Q.push(c);
}
dfs(Q.top(), "");
for_each(s, s + n, print);
}
return ;
}

Haffman编码(haffman树)的更多相关文章

  1. ACM Haffman编码

    Haffman编码 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 哈弗曼编码大家一定很熟悉吧(不熟悉也没关系,自己查去...).现在给你一串字符以及它们所对应的权值 ...

  2. java实现Haffman编码

    1.先创建一个树节点类(泛型类),为了方便使用集合的排序方法,泛型类要实现泛型接口Comparable,代码如下 package com.hjp.huffman; /** * Created by J ...

  3. Haffman编码

    Huffman树又称为最优树,是一种带权路径最短的树. 一.带权路径 在一棵树中我们把一个节点到另一个节点之间的通路叫做路径,在路径中每经过一个节点路径的长度就加一.如果对一个节点附上权值,则该节点的 ...

  4. 【算法】Prüfer编码 —— HNOI2004树的计数

    的确,如果不知道这个编码的话的确是一脸懵逼.在这里放一篇认为讲的很详细的 BLOG,有关于编码的方式 & 扩展在里面都有所提及. 欢迎点此进入 --> 大佬的博客 在这里主要想推导一下最 ...

  5. 4.15 省选模拟赛 编码 trie树 前缀和优化建图 2-sat

    好题 np. 对于20分 显然可以爆搜. 对于50分 可以发现每个字符串上的问号要么是0,要么是1.考虑枚举一个字符串当前是0还是1 这会和其他字符串产生矛盾. 所以容易 发现这是一个2-sat问题. ...

  6. 文件压缩小项目haffman压缩

    文件压缩的原理: 文件压缩总体可以分为有损压缩和无损压缩两类,有损压缩是指对mp3等格式的文件,忽略一些无关紧要的信息,只保留一些关键的信息,但并不因此影响用户对于这些mp3格式文件的体验度,无损压缩 ...

  7. deflate树与deflate编码

    关于deflate树,能搜到的资料非常少,这个概念来自gzip的压缩算法,是由huffman树转变过来的.这里简单记录下deflate树的生成过程以及deflate编码. 假设以5 8 9 10 14 ...

  8. 【转】ACM 2567 -- 树的Prufer编码

    本文介绍北京大学ACM网站2567号题目的解法.介绍部分基本翻译自网站上的题目介绍. 题目介绍:    给定一棵各节点编号为整数1,2,3...n的树(例如,无环连通图),其Prufer编码(Pruf ...

  9. 树的计数 Prüfer编码与Cayley公式 学习笔记

    最近学习了Prüfer编码与Cayley公式,这两个强力的工具一般用于解决树的计数问题.现在博主只能学到浅层的内容,只会用不会证明. 推荐博客:https://blog.csdn.net/moreja ...

随机推荐

  1. Arduino红外遥控系列教程2013——红外转码

    教程三:红外转码教程——用电视遥控器取代机顶盒遥控器 前言前段时间看到了使用红外遥控的X-Bot机器人[链接],感觉很有意思.最近开始玩Arduino与红外,一方面打算将来用于BOXZ的之间的通讯控制 ...

  2. MVC与WebForm最大的区别

    原文地址:http://www.cnblogs.com/birdshover/archive/2009/08/24/1552614.html 使用ASP.NET MVC框架,创建默认项目,第一直观感觉 ...

  3. poj 3287 The Trip, 2007_贪心

    题意:把一个包放入另一个包内,使得总共要带的件数最少,就是说大包可以装小包,且一个大包只能装一个小包,但是这个小包可以继续装更小的包. 思路:因为相同大小的包不能互相装,所以最小数量就是有相同尺寸的包 ...

  4. 实际用户ID,有效用户ID和设置用户ID

    摘自http://blog.csdn.net/guosha/article/details/2679334 实际用户ID,有效用户ID和设置用户ID 看UNIX相关的书时经常能遇到这几个概念,但一直没 ...

  5. C# 泛型转换 将object[] 数组转换为泛型列表

    public List<TaskSoftInfo> ReadSoftDownTaskList()        {                string[] Tid = (strin ...

  6. hdu 4753 Fishhead’s Little Game

    状态压缩dp解博弈问题(记忆化搜索).比赛的时候最后才开始做这道题,而且当时不知道为什么一直犯一些很2B的问题,导致没能ac,晚上看了看原先的代码,改了一下就MLE了...我原先是开的dp[1 < ...

  7. Linux学习2——文件与目录

    一.写在前面  在本节将介绍Linux下文件与目录的一些基本概念以及一些基本操作. 二.完成目标 1.了解文件和目录的一些基本概念 2.操作文件和目录的相关命令 3.文件内容查阅命令 4.文件查询命令 ...

  8. csss3 2D转换

    CSS3 转换 通过 CSS3 转换,我们能够对元素进行移动.缩放.转动.拉长或拉伸. 它如何工作? 转换是使元素改变形状.尺寸和位置的一种效果. 您可以使用 2D 或 3D 转换来转换您的元素. 浏 ...

  9. RequiredFieldValidator验证下拉列表框

    <asp:DropDownList ID="DropDownList1" runat="server" CssClass="style01&qu ...

  10. android开发网络连接工具类(一)

    网络连接工具类整理: package com.gzcivil.utils; import java.io.IOException; import java.util.ArrayList; import ...