【头歌】构建哈夫曼树及编码

构建哈夫曼树及编码

第1关:构建哈夫曼树

任务描述

本关任务:构建哈夫曼树,从键盘读入字符个数n及这n个字符出现的频率即权值,构造带权路径最短的最优二叉树(哈夫曼树)。

相关知识

哈夫曼树的定义

设二叉树具有n个带权值的叶子结点{w1,w2,...,wn},从根结点到每个叶子结点都有一个路径长度。

从根结点到各个叶子结点的路径长度与相应结点权值的乘积的和称为该二叉树的带权路径长度,记作:

其中,wi为第i个叶子结点的权值,li为第i个叶子结点的路径长度。

例如: 哈夫曼树

以上二叉树的带权路径长度值: WPL=1×3+3×3+5×2+7×1=29

给定一组具有确定权值的叶子结点,可以构造出许多形状的二叉树,把其中具有最小带权路径长度的二叉树称为哈夫曼树。

例如,用4个整数1、3、5、7作为4个叶子结点的权值,可以构造出不同的二叉树,它们的带权路径长度可能不相同,如下:

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
哈夫曼树是一种权路径长度最短的树,可以用于数据压缩等领域。构建哈夫曼树的过程可以通过递归实现,具体步骤如下: 1. 将所有权值存储在一个数组中,并按照从小到大的顺序排序。 2. 取出权值最小的两个节点,将它们合并成一个新节点,新节点的权值为两个节点的权值之和。 3. 将新节点插入到数组中,并按照权值从小到大的顺序排序。 4. 重复步骤2和3,直到数组中只剩下一个节点,即为哈夫曼树的根节点。 下面是C++实现哈夫曼树的代码示例: ```cpp #include <iostream> #include <queue> #include <vector> using namespace std; // 定义哈夫曼树节点结构体 struct HuffmanNode { int weight; // 权值 HuffmanNode* left; // 左子节点 HuffmanNode* right; // 右子节点 HuffmanNode(int w) : weight(w), left(nullptr), right(nullptr) {} }; // 定义比较函数,用于优先队列中节点的排序 struct cmp { bool operator()(HuffmanNode* a, HuffmanNode* b) { return a->weight > b->weight; } }; // 构建哈夫曼树 HuffmanNode* buildHuffmanTree(vector<int>& weights) { priority_queue<HuffmanNode*, vector<HuffmanNode*>, cmp> pq; for (int w : weights) { pq.push(new HuffmanNode(w)); } while (pq.size() > 1) { HuffmanNode* left = pq.top(); pq.pop(); HuffmanNode* right = pq.top(); pq.pop(); HuffmanNode* parent = new HuffmanNode(left->weight + right->weight); parent->left = left; parent->right = right; pq.push(parent); } return pq.top(); } // 生成哈夫曼编码 void generateHuffmanCode(HuffmanNode* root, string code, vector<string>& codes) { if (!root) { return; } if (!root->left && !root->right) { codes.push_back(code); return; } generateHuffmanCode(root->left, code + "0", codes); generateHuffmanCode(root->right, code + "1", codes); } int main() { vector<int> weights = {5, 9, 12, 13, 16, 45}; HuffmanNode* root = buildHuffmanTree(weights); vector<string> codes; generateHuffmanCode(root, "", codes); for (int i = 0; i < weights.size(); i++) { cout << "Weight: " << weights[i] << ", Code: " << codes[i] << endl; } return 0; } ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

大数据秃头族

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值