c语言哈夫曼编码 贪心,[C++]哈夫曼树(最优满二叉树) / 哈夫曼编码(贪心算法)

一 哈夫曼树

1.1 基本概念

算法思想

贪心算法(以局部最优,谋求全局最优)

适用范围

1 【(约束)可行】:它必须满足问题的约束

2 【局部最优】它是当前步骤中所有可行选择中最佳的局部选择

3 【不可取消】选择一旦做出,在算法的后面步骤中,就无法再改变。

示例

【树论:最优(二叉)数=带权路径最短的树】

【图论:单源最短路径=从某一结点出发至其他结点的带权路径之和最小 / 多源最短路径=某一对结点之间的带权路径之和最小】

定义

哈夫曼树 = 最优((满)二叉)树 = 带权路径之和最短的树

哈夫曼树所产生的哈夫曼编码 = (最优)前缀编码

一种最基本的、数据压缩的编码方法

5e7d5ebde7a484eb71cf66945f103bea.png

1.2 算法描述/构造过程

0029821a38758ecde31d20c5fb816f95.png

1.3 算法实现

8c926638c469691ef0d9e1f2bf1070f5.png

1> 定义存储结构

# define MAXNEGATIVE -9999 //最大负值

# define MAXPOSITIVE 9999 //最大正值

typedef struct HuffmanNode{ // 加 typedef : 避免发生编译错误 " variable or field 'functionName' declared void "

double weight;

int parent, lchild, rchild; //当前节点的双亲结点、左右孩子结点

}*HuffmanTree;

2> 确定最小的两个结点的选择策略(贪婪策略)【次核心】

/**

* 选择权值结点最小的两个结点 s1、s2

*/

void Select(HuffmanTree &HT,int k,int *s1,int *s2){

int min[2]={0, 0}; // 升序排名, 保留 HT中 权重值最小的两个结点的下标

cout<

HT[0].weight = MAXPOSITIVE; // 初始化为最大正值,方便被 初始结点比最小而比下去

for(int i=1;i<=k;i++){

//cout<

if( HT[i].parent==0 && (HT[i].weight

if(HT[i].weight

min[1] = min[0]; min[0] = i; cout<

} else {

min[1] = i;cout<

}

}

}

*s1 = min[0]; // 最小者下标

*s2 = min[1]; // 次小者下标

//cout<

}

3> 创建 Huffman 树【核心】

/**

* 创建 Huffman 树

* n : 初试结点(待编码结点)数

*/

void CreateHuffmanTree(HuffmanTree &HT, int n){

int m = 2*n-1;

HT = new HuffmanNode[m+1]; // 0 位空余

cout<

for(int i=1;i<=n;i++){

cin>>HT[i].weight;

}

for(int k=1;k<=m;k++){ // 初始化所有结点

HT[k].lchild = 0;

HT[k].rchild = 0;

HT[k].parent = 0;

}

for(int j=n+1;j<=m;j++){

int s1,s2;

Select(HT, j-1, &s1, &s2); // 选择权值结点最小的两个结点s1、s2

HT[s1].parent = j; // 合并左右结点

HT[s2].parent = j;

HT[j].lchild = s1;

HT[j].rchild = s2;

HT[j].weight = HT[s1].weight + HT[s2].weight;

}

}

4> 输出哈夫曼树

void OutputHuffmanTree(HuffmanTree &HT, int n){ // 输出 哈夫曼树 中 结点情况

int m = 2*n-1;

for(int i=1;i<=m;i++){

cout< weight: "<

cout<

cout<

}

}

5> 执行:Main函数

int main(){

HuffmanTree HT;

int n;

cout<

int s1, s2;

cin>>n;

CreateHuffmanTree(HT, n);

OutputHuffmanTree(HT, n);

return 0;

}

2814dfbccc08845889e0722cf32ddd5e.png

二 哈夫曼编码

2.1 哈夫曼编码的主要思想

0445876c508898a81946e9709c5f94d5.png

2.2 【前缀编码】

e4eb2562a6f6a84aacefe51f8d1e9515.png

2.3 【哈夫曼编码】

33c4b807992646944fc7c7765d0a62ba.png

2.4 应用:文件的编码与译码

e4970a485c6244986d647e63a8cbdc67.png

2.5 哈夫曼编码的算法实现 (根据哈夫曼树,求哈夫曼编码)

9526e890a15f0a862d0c92fdc6a17fba.png

1> 定义存储结构

//由于 每个哈夫曼编码是变长编码;因此,使用指针数组来存储每个字符编码串的首地址

typedef char **HuffmanCode; // 动态分配数组,存储哈夫曼编码表 (char 前面不能随意加 struct 关键字)

2> 生成哈夫曼编码(表)(借助于哈夫曼树)

/**

* 根据哈夫曼树,求哈夫曼编码

*/

void CreateHuffmanCodeByHuffmanTree(HuffmanTree HT, HuffmanCode &HC,int n){//从叶子到根,逆向求每个字符的哈夫曼编码

HC = new char *[n+1]; // 分配存储n个字符编码的编码表空间

char *cd = new char [n]; // 分配临时存放每个字符编码的动态数组空间

cd[n-1] = ' '; // 编码结束符

for(int i=1;i<=n;i++){ // 逐个字符求哈夫曼编码

int start = n-1; // start 初始化指向最后位置 即 编码结束符位置,逆向存储 从叶子到根结点路径的0/1编码

int c = i;

int f = HT[i].parent; // f 指向结点C 的双亲结点

while(f!=0){ // 直到 f 滑动到根节点为止

--start; // 编码表游标向前滑动

if(HT[f].lchild == c){

cd[start] = '0';

} else {

cd[start]= '1';

}

c = f;

f = HT[f].parent; // 逆向, 自(树)底向(树)上

}

HC[i] = new char [n-start]; //为第i个字符编码分配存储空间

strcpy(HC[i], &cd[start]); // 头文件: #include 、

}

delete cd;

}

3> 输出哈夫曼编码(表)

void OutputHuffmanCodes(HuffmanCode &HC,int n){

for(int i=1;i<=n;i++){

cout<

for(int j=0,size=sizeof(HC[i])/sizeof(char);j

cout<

}

cout<

}

}

4> 执行:Main函数

int main(){

HuffmanTree HT;

int n;

cout<

int s1, s2;

cin>>n;

CreateHuffmanTree(HT, n);

OutputHuffmanTree(HT, n);

HuffmanCode HC;

CreateHuffmanCodeByHuffmanTree(HT, HC, n);

OutputHuffmanCodes(HC, n);

return 0;

}

13842a61885ccc3257f15e9d037510fb.png

三 哈夫曼译(解)码

四 文献

4.1 参考文献

《数据结构(C语言版):严蔚敏/吴伟民》

《算法设计与分析基础([美] Anany Levitin. 潘彦 译)》

贪心算法思想的适用范围

4.2 推荐文献

五 二叉树 补充:红黑树

55c38250e3f52c407e132653f5f2f6bc.png

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值