考研系列之数据结构——哈夫曼树以及哈夫曼编码

//哈夫曼树结点结构
typedef struct HNode
{
	char data; //数据,比如说a,b,c... 
	double weight;//权重
	int parent;//双亲,-1表示没有双亲,即根节点
	int lchild;//左孩子,数组下标,-1表示无左孩子,即叶节点
	int rchild;//右孩子
}Hnode;
//编码结构,每个节点数据都对应一个编码 
typedef struct HCNode
{
	char data; //数据
	string code;//该字符编码
}Hcnode;
struct minnodes
{
	int m1;//两者更小权值结点下标
	int m2;//两者中权值更大的下表标
	bool flag;//若找到则为true,否则为false,false说明仅有一个结点
};
//辅助标志数组 标记该结点为根的树是否已加入哈夫曼树
bool flag[MAX] = { false };
Hnode HT[MAX];//哈夫曼树
Hcnode HC[MAX];//哈夫曼编码数组,用来防止代码冗余			   
			   //选择两棵最小权值的树 参数max,当前有权值结点下标+1
			
minnodes Select(int max)//用来选择两个权值最小的节点来组成哈夫曼树
{
	double min = MAXW;
	minnodes mins;
	mins.m2 = -1;
	//查找第一个最小权值的结点下标
	for (int i = 0; i < max; i++)
	{
		if (!flag[i] && HT[i].weight < min)//未加入哈夫曼树,权值更小
		{
			min = HT[i].weight;//更新最小权值
			mins.m1 = i;
		}
	}
	flag[mins.m1] = true;//将结点加入哈夫曼树
	min = MAXW;
	//查找第二个最小权值结点下标,可能不存在
	for (int i = 0; i < max; i++)
	{
		if (!flag[i] && HT[i].weight < min)//未加入哈夫曼树,权值更小
		{
			min = HT[i].weight;//更新最小权值
			mins.m2 = i;
		}
	}
	flag[mins.m2] = true;//将结点加入哈夫曼树
	if (-1 == mins.m2)//仅剩余一个结点未加入哈夫曼树
	{
		mins.flag = false;//未找到两棵最小权值树
	}
	else
	{
		mins.flag = true;
	}
	return mins;
}
/*
Project:哈夫曼树 构造 编码 译码 计算wpl
Date:    2019/02/04
Author:  Frank Yu
void CreateHT()创建哈夫曼树
void Code() 哈夫曼树编码
void Encode() 哈夫曼树解码
void WPL() 计算带权路径长度
*/
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<string>
#include<set>
#include<list>
#include<vector>
#include<map>
#include<iterator>
#include<algorithm>
#include<iostream>
#define MAX 1000 //哈夫曼树最大结点个数
#define MAXW 1000 //权值最大
using namespace std;
//哈夫曼树结点结构
typedef struct HNode
{
	char data; //数据,比如说a,b,c... 
	double weight;//权重
	int parent;//双亲,-1表示没有双亲,即根节点
	int lchild;//左孩子,数组下标,-1表示无左孩子,即叶节点
	int rchild;//右孩子
}Hnode;
//编码结构,每个节点数据都对应一个编码 
typedef struct HCNode
{
	char data; //数据
	string code;//该字符编码
}Hcnode;
//两个最小结点下标
struct minnodes
{
	int m1;//两者更小权值结点下标
	int m2;//两者中权值更大的下表 
	bool flag;//若找到则为true,否则为false,false说明仅有一个结点
};
bool flag[MAX] = { false };//辅助标志数组 标记该结点为根的树是否已加入哈夫曼树
Hnode HT[MAX];//哈夫曼树结构数组, 
Hcnode HC[MAX];//哈夫曼编码数组	,		   
			   //选择两棵最小权值的树 参数max,当前有权值结点下标+1
minnodes Select(int max)
{
	double min = MAXW;//用权值最大的节点来初始化 
	minnodes mins;//定义一个编码结构 
	mins.m2 = -1;
	//查找第一个最小权值的结点下标
	for (int i = 0; i < max; i++)
	{
		if (!flag[i] && HT[i].weight < min)//未加入哈夫曼树,如果权值更小
		{
			min = HT[i].weight;//更新最小权值
			mins.m1 = i;//将数据中较小 节点的下表更新为第i个节点 
		}
	}
	flag[mins.m1] = true;//将结点加入哈夫曼树之后,将标志改为true; 
	min = MAXW;//将最小值初试为最大值继续寻找第二个较小的节点 
	//查找第二个最小权值结点下标,可能不存在
	for (int i = 0; i < max; i++)
	{
		if (!flag[i] && HT[i].weight < min)//未加入哈夫曼树,权值更小的话 
		{
			min = HT[i].weight;//更新最小权值
			mins.m2 = i;//更新第二小节点的下标 
		}
	}
	flag[mins.m2] = true;//将结点加入哈夫曼树,标志改为true; 
	
	if (-1 == mins.m2)//仅剩余一个结点未加入哈夫曼树,也就是说没有找到第二个最小的节点, 
	{
		mins.flag = false;//未找到两棵最小权值节点 
	}
	else
	{
		mins.flag = true;//找到两个最小权值节点 
	}
	return mins;
}
//打印哈夫曼树
void PrintHT(int max)
{
	cout << "下标\t" << "数据\t" << "权重\t" << "双亲\t" << "左孩子\t" << "右孩子" << endl;
	for (int i = 0; i < max; i++)
	{
		cout << i << "\t" << HT[i].data << "\t" << HT[i].weight << "\t" << HT[i].parent << "\t" << HT[i].lchild << "\t" << HT[i].rchild << endl;
	}
}
//打印编码
void PrintHC(int n)
{
	for (int i = 0; i < n; i++)
	{
		cout << HC[i].data << ":" << HC[i].code << endl;;
	}
}
//创建哈夫曼树
void CreateHT()
{
	int n;//字符个数,即哈夫曼树叶节点个数
	minnodes mins;//创建最小的节点结构 
	cout << "请输入字符个数:" << endl;
	cin >> n;
	cout << "请输入字符及权值:" << endl;
	for (int i = 0; i < n; i++)
	{
		cin >> HT[i].data >> HT[i].weight;
		HT[i].lchild = -1; HT[i].rchild = -1;//表示该节点还没有进行构建哈夫曼树 
	}
	/*
	HT[0].data = 'a';HT[0].weight = 45;HT[0].lchild = -1;HT[0].rchild = -1;
	HT[1].data = 'b';HT[1].weight = 13;HT[1].lchild = -1;HT[1].rchild = -1;
	HT[2].data = 'c';HT[2].weight = 12;HT[2].lchild = -1;HT[2].rchild = -1;
	HT[3].data = 'd';HT[3].weight = 16;HT[3].lchild = -1;HT[3].rchild = -1;
	HT[4].data = 'e';HT[4].weight = 9; HT[4].lchild = -1;HT[4].rchild = -1;
	HT[5].data = 'f';HT[5].weight = 5; HT[5].lchild = -1;HT[5].rchild = -1;*/
	int i = n;//
	for (;; i++)
	{
		mins = Select(i);//在这n个节点当中找到两棵根权值最小的树
		if (mins.flag == false)//仅剩余一棵树时跳出
		{
			HT[mins.m1].parent = -1;
			break;
		}
		HT[i].weight = HT[mins.m1].weight + HT[mins.m2].weight;//新加入哈夫曼树结点为两个结点权值之和
		HT[i].data = ' ';//此时新节点的数据为空,因为不涉及编码以及求wpl; 
		HT[mins.m1].parent = i; //两个权值最小结点双亲为新加入结点
		HT[mins.m2].parent = i;//两个节点的双亲 
		HT[i].lchild = mins.m1;//权值更小的节点成为左孩子 
		HT[i].rchild = mins.m2;//权值更大的节点成为右孩子 
	}
	PrintHT(i);//打印哈夫曼树
}
//哈夫曼树编码
void Code()
{
	int i = 0;
	for (;; i++)//给所有叶子结点编码
	{
		int j = i;
		string str = "";
		HC[i].data = HT[i].data;//复制数据
		while (-1 != HT[j].parent)//从叶节点找到根
		{
			if (HT[HT[j].parent].lchild == j)//左0右1
			{
				str += '0';
			}
			else
			{
				str += '1';
			}
			j = HT[j].parent;
		}
		reverse(str.begin(), str.end());//逆序
		HC[i].code = str;              //保存至编码
		if (HT[i].lchild == -1 && HT[i].rchild == -1)continue;//非叶子不编码
		else break;
	}
	PrintHC(i);
}
//哈夫曼树解码 从根开始,左0右1,直至叶节点
void Encode()
{
	string s;
	int root = 0;//记录根节点的下标
	cout << "请输入01字符串:" << endl;
	cin >> s;
	while (HT[root].parent != -1) root++;
	int j = root;
	for (int i = 0; i < s.length(); i++)//遍历输入的01串
	{
		if ('0' == s[i])
		{
			j = HT[j].lchild;
		}
		else
		{
			j = HT[j].rchild;
		}
		if (HT[j].lchild == -1 && HT[j].rchild == -1)//到达叶节点
		{
			cout << HT[j].data;
			j = root;//返回根节点继续
		}
	}
	cout << endl;
}
//计算WPL
void WPL()
{
	double WPL = 0;
	for (int i = 0;; i++)
	{
		if (HT[i].lchild != -1 || HT[i].rchild != -1)
		break;//这里只有叶子节点的才会有权值,所以要找到叶子节点 
		WPL += HT[i].weight * HC[i].code.length();//权值×路径长度(编码长度)	
	}
	cout << "WPL:" << WPL << endl;
}
//菜单
void menu()
{
	cout << "************1.创建哈夫曼树   2.编码************" << endl;
	cout << "************3.解码           4.计算wpl*********" << endl;
	cout << "************5.退出" << endl;
}
//主函数
int main()
{
	int choice = 0;
	while (1)
	{
		menu();
		printf("请输入菜单序号:\n");
		scanf("%d", &choice);
		if (5 == choice) break;
		switch (choice)
		{
		case 1:CreateHT(); break;
		case 2:Code(); break;
		case 3:Encode(); break;
		case 4:WPL(); break;
		default:printf("输入错误!!!\n"); break;
		}
	}
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
哈夫曼树是一种最优二叉树,它的构造算法如下:首先将n个结点作为n棵仅含有一个根结点的二叉树,构成一个森林F。然后,生成一个新结点,并从F中找出根结点权值最小的两棵树作为它的左右子树,新结点的权值为两棵子树根结点的权值之和。接着,从F中删除这两个树,并将新生成的树加入到F中。重复以上步骤,直到F中只有一棵树为止。 哈夫曼树具有以下性质: 1)每个初始结点都会成为叶节点,双支结点都为新生成的结点 2)权值越大离根结点越近,反之权值越小离根结点越远 3)哈夫曼树中没有结点的度为1 4)n个叶子结点的哈夫曼树的结点总数为2n-1,其中度为2的结点数为n-1。 哈夫曼树的应用之一是解决编码问题。在编码中,我们使用二进制来表示字符,其中固定长度编码和前缀编码是两种常见的编码方式。前缀编码要求没有一个编码是另一个编码的前缀,而哈夫曼树构造的编码正是满足前缀编码的要求。 所以,哈夫曼树数据结构考研中是一个重要的概念,它可以用于构造最优二叉树和解决编码问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [数据结构考研笔记(十六) ——哈夫曼树、编码应用](https://blog.csdn.net/sf9090/article/details/109154652)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值