C++实现哈夫曼树的方法

 更新时间:2020年04月28日 11:29:29   作者:李杨在路上  
这篇文章主要为大家详细介绍了C++实现哈夫曼树的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

序言

对于哈夫曼编码,个人的浅薄理解就是在压缩存储空间用很大用处。
用一个很简单例子,存储一篇英文文章时候,可能A出现的概率较大,Z出现的记录较小,如果正常存储,可能A与Z存储使用的空间一样。但是用哈夫曼编码方式,A经常出现,所用编码长度就短。

构造哈夫曼树,生成哈夫曼编码

一、定义节点类型

struct Node {
 char C;
 long key;
 Node *Left, *Right,*parent;
 Node() { Left = Right = NULL; }
};

二、定义树类型(节点数组)

三要素:不定长数组,元素大小,有效元素个数

struct RootA {
 Node *NodeA;
 const int Size;
 int n;
 RootA(int Size) :Size(Size) { n = 0; NodeA = new Node[Size]; }
 ~RootA() { delete[]NodeA; }
};

三、创建哈夫曼树

1.将每一个节点都当成一棵树,初始化数组大小,并进行赋值

RootA RA(4);
 //1.在RA.NodeA中存入字母和权值
 for (RA.n = 0;RA.n < RA.Size;RA.n++) {
 cout << "字母:";
 cin >> RA.NodeA[RA.n].C;
 cout << "权值:";
 cin >> RA.NodeA[RA.n].key;
 }

2.将树按权值大小排序

void Sort(RootA *ra) {
 for (int i = 0;i < ra->n;i++) {
 bool ESC = false;
 for (int j = 0;j < ra->n - i - 1;j++) {
  if (ra->NodeA[j].key > ra->NodeA[j + 1].key) {
  Node T;T = ra->NodeA[j];ra->NodeA[j] = ra->NodeA[j + 1];ra->NodeA[j + 1] = T;
  ESC = true;
  }
 }
 if (!ESC) return;
 }
}

3.(1)遍历数组,将RA.NodeA[0]和RA.Node[1]合并,其余向前移动,重新排序
(2)将RA.NodeA[0],RA.NodeA[1]分别放在新合并的RA.NodeA[0]的左右子结点中

while (RA.n > 1) {
 //1.将RA.NodeA[0]和RA.NodeA[1]合并,将其余向前移动
 Node *NewNode0 = new Node;
 *NewNode0 = RA.NodeA[0];
 Node *NewNode1 = new Node;
 *NewNode1 = RA.NodeA[1];
 RA.NodeA[0].C = ' ';
 RA.NodeA[0].key = RA.NodeA[0].key + RA.NodeA[1].key;
 RA.NodeA[0].Left = NewNode0;
 NewNode0->parent = &RA.NodeA[0];
 RA.NodeA[0].Right = NewNode1;
 NewNode1->parent = &RA.NodeA[0];
 for (int i = 1;i < RA.n-1;i++) {
  RA.NodeA[i] = RA.NodeA[i + 1];
 }
 RA.n = RA.n - 1;
 //2.排序
 Sort(&RA);
 }

4.输出哈夫曼编码

递归,找到叶子节点,记录路径,左记录0,右记录1,直到输出所有叶子节点

void CrateCode(Node *t,string &s) {
 //1.遍历节点,遍历左节点编码为0,右节点则为1,递归,直到输出所有叶子节
 if (t->Left != NULL && t->Right != NULL) {
 s.push_back('0'); CrateCode(t->Left, s);
 s.pop_back();
 s.push_back('1');CrateCode(t->Right, s);
 s.pop_back();
 }
 else {
 cout << "哈夫曼编码:";
 cout << t->C << ":" << s<<endl;
 }
}

以上是对构造哈夫曼树以及生成哈夫曼编码的总结,希望对你们有所帮助!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C++ cin输入的多种方法详解

    C++ cin输入的多种方法详解

    cin是C++编程语言中的标准输入流对象,即stream类的对象。cin主要用于从标准输入读取数据,这里的标准输入,指的是终端的键盘。接下来通过本文给大家分享C++ cin输入的几种方式,一起看看吧
    2021-09-09
  • C/C++ Qt 数据库QSql增删改查组件应用教程

    C/C++ Qt 数据库QSql增删改查组件应用教程

    Qt SQL模块是Qt中用来操作数据库的类,该类封装了各种SQL数据库接口,可以很方便的链接并使用。本文主要介绍了Qt数据库QSql增删改查组件的应用教程,感兴趣的同学可以学习一下
    2021-12-12
  • C语言 OutputDebugString与格式化输出函数OutputDebugPrintf案例详解

    C语言 OutputDebugString与格式化输出函数OutputDebugPrintf案例详解

    这篇文章主要介绍了C语言 OutputDebugString与格式化输出函数OutputDebugPrintf案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • 深入理解Qt中各种消息框对话框的使用

    深入理解Qt中各种消息框对话框的使用

    本篇文章主要介绍了Qt中各种消息框的使用,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-07-07
  • C语言实现三子棋小游戏全程详解

    C语言实现三子棋小游戏全程详解

    完成一个三子棋的代码并不是很难,有困难且重要的是完成这个游戏代码所具备的思想,因为思想上的进步才是真正的进步,当我们有了这个思想上的武器,写出别的代码,难度就不会高
    2022-05-05
  • udp socket客户端和udp服务端程序示例分享

    udp socket客户端和udp服务端程序示例分享

    这篇文章主要介绍了udp socket客户端和udp服务端程序示例,需要的朋友可以参考下
    2014-03-03
  • C/C++ Qt TreeWidget 嵌套节点操作使用

    C/C++ Qt TreeWidget 嵌套节点操作使用

    本文主要介绍了TreeWidget的如何使用,实现对树形框多节点的各种操作,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • C语言实现学生信息管理系统(文件操作)

    C语言实现学生信息管理系统(文件操作)

    这篇文章主要介绍了C语言实现学生信息管理系统,增加了文件操作,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • C++交换指针实例

    C++交换指针实例

    这篇文章主要介绍了C++交换指针实例,针对C与C++交换指针的方法进行了较为详细的对比分析,非常具有实用价值,需要的朋友可以参考下
    2014-10-10
  • 基于MFC实现自定义复选框效果

    基于MFC实现自定义复选框效果

    复选框是一种可同时选中多项的基础控件,主要是有两种明显的状态:选中与非选中。本文将通过MFC框架实现自定义复选框效果,感兴趣的可以了解一下
    2022-02-02

最新评论