哈夫曼树(Huffmantree)

article/2025/8/25 13:26:31

1.基本概念

  哈夫曼树又称为最优树,是一类带权路径长度最短的树。

一些概念的定义:

(1)路径:树的两个结点之间的连线称为路径。

(2)路径长度:路径上的分支数目称作路径长度。若规定根结点长度为1,则从根结点到第L层结点的路径长度为L-1。

(3)权:是对实体的某个或某个属性的数值化描述,可分为结点权和边权。如下:

如图所示,根结点权值为1,根结点的左子树结点权值为2,若两节点之间的连线有值,称为边权。

(4)结点的带权路径长度:从根结点到该结点之间的路径长度为该结点的权的乘积。

(5)树的带权路径长度:树中所有叶子结点的带权路径长度之和,通常记作WLP=\sum_{k=1}^{n}w*k

 w为结点值,k为层数。

给定N个权值作为N个叶子结点,构造一颗二叉树,若该树的带权路径长度达到最小,称这样的二叉树为最优二叉树,也称为哈夫曼树。带权路径长度最短的树,权值较大的结点离根较近。

2.哈夫曼树的构造

(1).给定n个权值,这n棵二叉树构成森林F。

(2)在森林F中选取两棵根结点的权值最小的树作为左右子树构造一棵新的二叉树,且置新的二叉树的根结点的权值为其左右子树上根结点的权值之和。

(3)在森林F中删除这两棵树,同时将新得到的二叉树加入F中。

 

 (4)重复(2)(3)步骤,直到F中只含有一棵树为止。

在构造哈夫曼树时候,首先选择权值小的,这样保证权值大的离根最近,这样计算带权路径长度为最小,这是一种典型的贪心法。

3.代码实现

1.一个结点需要包含的内容:

typedef struct
{int lchild,rchild,parent;  //结点的左右孩子以及父节点的下标int weight; //权值
}*haffmantree,htnode;

2.查找权值最小的两个结点的思想是:从树起始位置开始,首先找到两个无父结点的结点(说明还未使用其构建成树),然后和后续无父结点的结点依次做比较,有两种情况需要考虑:

  • 如果比两个结点中较小的那个还小,就保留这个结点,删除原来较大的结点;
  • 如果介于两个结点权重值之间,替换原来较大的结点;
    void select(haffmantree ht,int end,int *s1,int *s2)
    {int min1,min2;int i=1;while(ht[i].parent!=0&&i<=end)i++;min1=ht[i].weight;*s1=i;i++;while(ht[i].parent!=0&&i<=end)i++;if(ht[i].weight<min1){min2=min1;*s2=*s1;min1=ht[i].weight;*s1=i;}else{min2=ht[i].weight;*s2=i;}for(int j=i+1;j<=end;j++){if(ht[j].parent!=0)continue;if(ht[j].weight<min1){min2=min1;min1=ht[j].weight;*s2=*s1;*s1=j;}else if(ht[j].weight>=min1&&ht[j].weight<min2){min2=ht[j].weight;*s2=j;}}
    }
    

    3.构建哈夫曼树

  • void creat(haffmantree *hp,int n,int *sum)
    {if(n<=1) return ;int m=2*n-1;  //哈夫曼树总结点数量,n为叶子结点*hp=(haffmantree)malloc(sizeof(haffmantree)*m);haffmantree h=*hp;//初始化哈夫曼树中的结点for(int i=1;i<=m;i++){h[i].parent=0;h[i].lchild=0;h[i].rchild=0;}for(int i=1;i<=n;i++)scanf("%d",&h[i].weight);//构建for(int i=n+1;i<=m;i++){int s1,s2;select(h,i-1,&s1,&s2);  //找到森林中两个值最小的结点h[s1].parent=i;h[s2].parent=i;h[i].lchild=s1;h[i].rchild=s2;h[i].weight=h[s1].weight+h[s2].weight;//*sum+=h[i].weight;  //树的路径长度}
    }


http://chatgpt.dhexx.cn/article/GRKMyust.shtml

相关文章

哈夫曼树详解及其应用(哈夫曼编码)

一&#xff0c;哈夫曼树的基本概念 路径&#xff1a;从树中一个结点到另一个结点之间的分支构成这两个结点间的路径 结点的路径长度&#xff1a;两结点之间路径上的分支数 树的路径长度&#xff1a;从树根到每一个结点的路径长度之和&#xff0e;记作&#xff1a;&#xff3…

哈夫曼树编码的实现+图解(含全部代码)

目录 哈夫曼树的基本概念 ------------哈夫曼树的构造方法 ------------------------哈夫曼编码 ------------------------------------全部代码 哈夫曼树的基本概念 哈夫曼树通常以二叉树的形式出现&#xff0c;所以也称最优二叉树&#xff0c;是一类带权路径长度最短的树…

哈夫曼树(C语言实现)

文章目录 哈夫曼树的基本概念哈夫曼树的构建构建思路代码实现 哈夫曼编码的生成编码生成思路代码实现 完整代码展示以及代码测试 哈夫曼树的基本概念 在认识哈夫曼树之前&#xff0c;你必须知道以下几个基本术语&#xff1a; 1、什么是路径&#xff1f; 在一棵树中&#xff0c…

打开VS2010提示:产品密钥框

打开VS2010提示&#xff1a;产品密钥框&#xff0c;如下图&#xff1a; …

VS 2017 产品密钥

个人分类&#xff1a; vs2010 Visual Studio 2017&#xff08;VS2017&#xff09; 企业版 Enterprise 注册码&#xff1a;NJVYC-BMHX2-G77MM-4XJMR-6Q8QF Visual Studio 2017&#xff08;VS2017&#xff09; 专业版 Professional 激活码key&#xff1a;KBJFW-NXHK6-W4WJM-CRM…

vs++2010学习版的注册密钥

6VPJ7-H3CXH-HBTPT-X4T74-3YVY7 欢迎使用Markdown编辑器 你好&#xff01; 这是你第一次使用 Markdown编辑器 所展示的欢迎页。如果你想学习如何使用Markdown编辑器, 可以仔细阅读这篇文章&#xff0c;了解一下Markdown的基本语法知识。 新的改变 我们对Markdown编辑器进行…

存储过程入门

参考文章 Oracle Database concepts guide&#xff08;11g2&#xff09; By Thomas KyteStored Procedure Wiki 先修知识 数据库的基本概念SQL 什么是存储过程&#xff08;Stored Procedure&#xff09;&#xff1a; 一段存储在数据库的“子程序”&#xff0c;下面对这两个…

【MySQL存储过程】创建一个简单的存储过程

什么是存储过程和函数 存储过程和函数是在数据库中定义的一些SQL语句的集合&#xff0c;然后直接调用这些存储过程和函数来执行已经定义好的SQL语句。存储过程和函数可以避免开发人员重复编写相同的SQL语句。而且&#xff0c;存储过程和函数是在MySQL服务器中存储和执行的&…

什么是存储过程

什么是存储过程&#xff1a;存储过程可以说是一个记录集吧&#xff0c;它是由一些T-SQL语句组成的代码块&#xff0c;这些T-SQL语句代码像一个方法一样实现一些功能&#xff08;对单表或多表的增删改查&#xff09;&#xff0c;然后再给这个代码块取一个名字&#xff0c;在用到…

MySQL-存储过程

文章目录 存储过程一. 存储过程的创建和使用1. 创建存储过程2. 删除存储过程3. 查看存储过程4. 调用存储过程5. 例题 二. 变量1. 系统变量1.1 全局变量1.2 会话变量 2. 自定义变量2.1 用户变量2.2 局部变量 三. 存储过程参数3.1 说明&#xff1a;3.2 例题 四. 流程控制1. IF语句…

存储过程怎么使用

1.什么是存储过程&#xff1f; 存储过程是封装了一条或多条SQL的集合。它的好处是简单、高性能、安全。2.为什么要使用存储过程&#xff1f; 简化复杂的操作&#xff0c;把SQL封装起来容易使用。 如果所有开发人员和应用程序都使用同一存储过程&#xff0c;则所有使用的代码都…

SQL创建存储过程

创建SQL存储过程需要使用到的语法 - 创建存储过程 CREATE 存储过程的名称(参数) BEGIN ...需要执行的SQL语句 END- 调用 CALL 存储过程的名称(参数)个人看法&#xff0c;这就是一个函数...无参数 CREATE PROCEDURE p_student_select() BEGIN SELECT * FROM student; ENDCALL …

4.3.1 存储过程的简要介绍

4.3.1 存储过程的简要介绍 1、什么是存储过程&#xff1f; 存储过程是一种命名的PL/SQL代码块。它既可以没有参数&#xff0c;也可以有若干输入、输出参数&#xff0c;甚至可以有多个既作输入又作输出的参数&#xff0c;但他通常没有返回值。 存储过程被保存在数据库中&#x…

储存过程

储存过程是一组为了完成特定功能的SQL语句表&#xff0c;经过编译后储存在数据库中&#xff0c;用户通过指定过程的名字并给定参数来调用执行它。 从常用的操作数据库的SQL语句在执行的时候需要先编译&#xff0c;然后执行&#xff0c;储存过程&#xff0c;则是采用另外一种方式…

java笔试--北京轩宇信息

第一题 import java.io.IOException; import java.nio.file.Files; import java.nio.file.Paths;/*** <p>功能: 编写程序,在C盘根目录下创建文件myFile.txt&#xff0c;文件内容如下&#xff0c;请注意缩进和换行&#xff1a;* Java* C/C* Python* JavaScripts*/ public…

笔试——中兴

参考达尔文公众号&#xff1a;https://mp.weixin.qq.com/s?__bizMzg5MDIwNjIwMA&mid2247496018&idx1&snf8109b6f5b5ea3a175e52eb7074bb7bc&chksmcfe293c5f8951ad3570a64a07ce0deba1ec12f3c8d0a15bbf1c64ed25e5faca46ef5974fef72&mpshare1&scene23&…

中兴2016笔试题答案Java_中兴Java笔试题

中兴Java笔试题 一、选择题(每题4分,共80分) 1. 编译Java Application 源程序文件将产生相应的字节码文件&#xff0c;这些字节码文件的扩展名为( ) A. .java B. .class C. .html D. . 2. main方法是Java Application程序执行的入口点&#xff0c;关于main方法的方法头以下哪项…

中兴通讯2013校招软件笔试题

关于const的实现机制&#xff0c;请看&#xff1a; http://blog.csdn.net/syzcch/article/details/8182184 define宏定义那个题&#xff1a; http://zhidao.baidu.com/link?urltSvmJ_ytFjwWKBLzDgCfLfW-mdJtTChTab3XzBAbd2x1nGYQCGnqDq__9-dqc_ndlWE1uPeaFcyVXlKOn1CAha …

中兴笔试程序题

文本编辑器&#xff08;15&#xff09; 要求&#xff1a; &#xff08;1&#xff09;编辑文本&#xff1b; &#xff08;2&#xff09;保存、打开指定位置的文本文件&#xff1b; &#xff08;3&#xff09;具有输入输出界面。 代码&#xff1a;&#xff08;此代码在vc6.…

中兴2016校招软件在线笔试题

面试经验可以参考我的另一篇文章&#xff0c;是7月初参加openday面试的&#xff0c;文章链接http://blog.csdn.net/dandelion1314/article/details/47009585 招聘群里有人发的招聘时间安排&#xff0c;仅供参考。 据说今年是中兴的第一次在线笔试&#xff0c;摄像头监控&am…