使用C++代码实现哈夫曼树的构造

简介: 使用C++代码实现哈夫曼树的构造

哈夫曼树是一种用于数据压缩的树形数据结构,其构造过程可以通过以下步骤实现:

 

1. 定义哈夫曼树的节点结构体,包括权重和指向左右子节点的指针。

2. 创建一个优先队列(最小堆),用于存储权重最小的节点。

3. 将所有权重作为单独的节点插入优先队列。

4. 重复以下步骤,直到只剩下一个节点为止:

     a. 从优先队列中取出两个权重最小的节点作为左右子节点。

     b. 创建一个新节点,权重为两个子节点的权重之和,将这个新节点插入优先队列。

5. 最终剩下的节点即为哈夫曼树的根节点。

 

以下是一个简单的 C++ 实现哈夫曼树构造的示例代码:

 

```cpp
#include <iostream>
#include <queue>
 
using namespace std;
 
struct Node {
    char data;
    int weight;
    Node* left;
    Node* right;
 
    Node(char data, int weight) : data(data), weight(weight), left(nullptr), right(nullptr) {}
};
 
struct Compare {
    bool operator()(Node* a, Node* b) {
        return a->weight > b->weight;
    }
};
 
Node* buildHuffmanTree(priority_queue<Node*, vector<Node*>, Compare>& pq) {
    while (pq.size() > 1) {
        Node* left = pq.top();
        pq.pop();
 
        Node* right = pq.top();
        pq.pop();
 
        Node* parent = new Node('$', left->weight + right->weight);
        parent->left = left;
        parent->right = right;
 
        pq.push(parent);
    }
 
    return pq.top();
}
 
void printCodes(Node* root, string code) {
    if (root == nullptr) {
        return;
    }
 
    if (root->data != '$') {
        cout << root->data << ": " << code << endl;
    }
 
    printCodes(root->left, code + "0");
    printCodes(root->right, code + "1");
}
 
int main() {
    priority_queue<Node*, vector<Node*>, Compare> pq;
 
    pq.push(new Node('a', 5));
    pq.push(new Node('b', 9));
    pq.push(new Node('c', 12));
    pq.push(new Node('d', 13));
    pq.push(new Node('e', 16));
    pq.push(new Node('f', 45));
 
    Node* root = buildHuffmanTree(pq);
 
    cout << "Huffman Codes are: " << endl;
    printCodes(root, "");
 
    return 0;
}
```

 

在这个示例中,我们定义了一个 `Node` 结构体表示哈夫曼树的节点,以及一个比较器 `Compare` 用于优先队列中节点的比较。然后我们实现了 `buildHuffmanTree` 函数来构建哈夫曼树,以及 `printCodes` 函数来打印哈夫曼编码。

 

`main` 函数中,我们创建了一个优先队列,并插入了一些示例数据。然后构建哈夫曼树,并打印出对应的哈夫曼编码。

 

这个示例展示了如何使用 C++ 实现哈夫曼树的构造过程。

相关文章
|
28天前
|
C++
C++ 语言异常处理实战:在编程潮流中坚守稳定,开启代码可靠之旅
【8月更文挑战第22天】C++的异常处理机制是确保程序稳定的关键特性。它允许程序在遇到错误时优雅地响应而非直接崩溃。通过`throw`抛出异常,并用`catch`捕获处理,可使程序控制流跳转至错误处理代码。例如,在进行除法运算或文件读取时,若发生除数为零或文件无法打开等错误,则可通过抛出异常并在调用处捕获来妥善处理这些情况。恰当使用异常处理能显著提升程序的健壮性和维护性。
42 2
|
21天前
|
算法框架/工具 C++ Python
根据相机旋转矩阵求解三个轴的旋转角/欧拉角/姿态角 或 旋转矩阵与欧拉角(Euler Angles)之间的相互转换,以及python和C++代码实现
根据相机旋转矩阵求解三个轴的旋转角/欧拉角/姿态角 或 旋转矩阵与欧拉角(Euler Angles)之间的相互转换,以及python和C++代码实现
90 0
|
28天前
|
程序员 C++ 开发者
C++命名空间揭秘:一招解决全局冲突,让你的代码模块化战斗值飙升!
【8月更文挑战第22天】在C++中,命名空间是解决命名冲突的关键机制,它帮助开发者组织代码并提升可维护性。本文通过一个图形库开发案例,展示了如何利用命名空间避免圆形和矩形类间的命名冲突。通过定义和实现这些类,并在主函数中使用命名空间创建对象及调用方法,我们不仅解决了冲突问题,还提高了代码的模块化程度和组织结构。这为实际项目开发提供了宝贵的参考经验。
42 2
|
28天前
|
C++
拥抱C++面向对象编程,解锁软件开发新境界!从混乱到有序,你的代码也能成为高效能战士!
【8月更文挑战第22天】C++凭借其强大的面向对象编程(OOP)能力,在构建复杂软件系统时不可或缺。OOP通过封装数据和操作这些数据的方法于对象中,提升了代码的模块化、重用性和可扩展性。非OOP方式(过程化编程)下,数据与处理逻辑分离,导致维护困难。而OOP将学生信息及其操作整合到`Student`类中,增强代码的可读性和可维护性。通过示例对比,可以看出OOP使C++代码结构更清晰,特别是在大型项目中,能有效提高开发效率和软件质量。
20 1
|
22天前
|
C++
C++代码来计算一个点围绕另一个点旋转45度后的坐标
C++代码来计算一个点围绕另一个点旋转45度后的坐标
42 0
|
22天前
|
C++
Resharper c++ 使用Enter自动补全代码
Resharper c++ 使用Enter自动补全代码
28 0
|
29天前
|
监控 编译器 C++
【代码讲解】【C/C++】获取文件最后修改的时间(系统时间)
【代码讲解】【C/C++】获取文件最后修改的时间(系统时间)
33 0
|
2月前
|
前端开发 编译器 程序员
协程问题之为什么 C++20 的协程代码比其他语言的协程 demo 长很多如何解决
协程问题之为什么 C++20 的协程代码比其他语言的协程 demo 长很多如何解决
|
1月前
|
JavaScript 前端开发 Java
面向对象编程(C++篇2)——构造
面向对象编程(C++篇2)——构造
26 0
|
2月前
|
算法 NoSQL 编译器
如何编写可维护的C++代码
如何编写可维护的C++代码