哈夫曼树的构建(C语言)

算法思路

主要包括两部分算法,一个是在数组中找到权值最小、且无父结点两个结点位置,因为只有无父结点才能继续组成树; ​ 另一个就是根据这两个结点来修改相关结点值。

  1. 结构定义和头文件

     
     #include <stdio.h>
    #include <malloc.h>
    #include <stdlib.h>
    #include <string.h>

    #define OVERFLOW -1

    typedef struct {
    int weight;//结点权值
    int lchild, rchild, parent;//结点左、右孩子、父结点
    }HNode,*HTree;
  2. 在数组中找到目前权值最小的两个结点 由于哈夫曼树规定结点左子树权值小于右子树,所以我这里把权值较小的那个结点位置赋给p1

    这部分我先找到前两个无父结点的结点位置赋给p1和p2,再继续遍历之后的与当前的p1和p2位置的结点权值比较

    • 若结点有父结点,直接跳过

    • 若结点无父结点,且权值小于p1,则将该位置赋给p1,令p2等于之前的p1

    • 若结点无父结点,且权值大于p1、小于p2,则将该位置赋给p2

      void selectMin(HTree HT,int length, int* p1, int* p2) {//搜索当前数组中无父结点的权值最小的两个结点的下标
    int i = ;//数组下标从1开始,0不用

    while (HT[i].parent!= && i <= length)//遍历到第一个无父结点的结点位置
    i++;
    *p1 = i; i++;
    while (HT[i].parent!= && i <= length)//遍历到第二个无父结点的结点位置
    i++;
    *p2 = i; i++;

    if (HT[*p1].weight > HT[*p2].weight) {//令p1始终指向较小权值的结点位置
    int temp = *p1;
    *p1 = *p2;
    *p2 = temp;
    }

    for (int n = i; n <= length; n++) {//继续遍历,比较之后的无父结点的结点权值与p1、p2
    if (HT[n].parent != )//若该结点有父结点,直接跳过
    continue;
    else if (HT[n].weight < HT[*p1].weight) {//若该结点权值小于p1,令p1等于n,p2等于p1
    *p2 = *p1;
    *p1 = n;
    }
    else if (HT[n].weight > HT[*p1].weight&& HT[n].weight < HT[*p2].weight)//若该结点权值大于p1,小于p2,令*p2=n
    *p2 = n;
    }

    return;
    }
  3. 构建哈夫曼树

      void createHuffmanTree() {//构建哈夫曼树
    int lnode;//哈夫曼树叶子结点数
    printf("input leafnode number:");
    scanf_s("%d", &lnode);
    int length=*lnode-;//哈夫曼树结点数=2*叶子节点数-1

    HTree HT = (HTree)malloc(sizeof(HNode) * (length + ));//数组下标从1开始,所以分配(length+1)大小空间
    if (!HT) exit(OVERFLOW);
    memset(HT, , sizeof(HNode) * (length + ));//将数组内元素都初始化为0

    HTree p = HT;
    for (int i = lnode + ; i <=length; i++) {//把所有非叶子节点的结点权值规定为无穷大,否则会影响接下来选择结点最小值
    (p + i)->weight = ;
    }


    printf("input leafnode weight:");
    for (int i = ; i <= lnode; i++) {//输入叶子结点权值
    scanf_s("%d", &(p+i)->weight);
    }

    int p1, p2;
    for (int i = lnode+; i <= length; i++) {//从第一个非叶子结点开始遍历
    selectMin(p, length, &p1, &p2);
    (p + i)->lchild = p1;//修改左子树的值
    (p + i)->rchild = p2;//修改左子树的值
    (p + i)->weight = (p + p1)->weight + (p + p2)->weight;//修改权值
    (p + p1)->parent = i;//修改左子树的父结点值
    (p + p2)->parent = i;//修改右子树的父结点值
    }

    for (int i = ; i <= length; i++) {
    printf("%3d %3d %3d %3d %3d\n", i, (p + i)->weight, (p + i)->parent, (p + i)->lchild, (p + i)->rchild);//遍历输出
    }

    return;
    }
  4. 主函数及具体实例

      int main() {

    createHuffmanTree();
    return ;
    }

自我总结

写这部分时候动态数组分配和使用那里耗了点时间,主要原因还是不太熟以及vs的操作太迷了。经过这次训练又加深了一点理解,vs的调试也逐渐能够熟练使用了,还是挺开心的。

哈夫曼树的构建(C语言)的更多相关文章

  1. 哈夫曼树(一)之 C语言详解

    本章介绍哈夫曼树.和以往一样,本文会先对哈夫曼树的理论知识进行简单介绍,然后给出C语言的实现.后续再分别给出C++和Java版本的实现:实现的语言虽不同,但是原理如出一辙,选择其中之一进行了解即可.若 ...

  2. 【算法】赫夫曼树(Huffman)的构建和应用(编码、译码)

    参考资料 <算法(java)>                           — — Robert Sedgewick, Kevin Wayne <数据结构>       ...

  3. Android版数据结构与算法(七):赫夫曼树

    版权声明:本文出自汪磊的博客,未经作者允许禁止转载. 近期忙着新版本的开发,此外正在回顾C语言,大部分时间没放在数据结构与算法的整理上,所以更新有点慢了,不过既然写了就肯定尽力将这部分完全整理好分享出 ...

  4. 【数据结构】赫夫曼树的实现和模拟压缩(C++)

    赫夫曼(Huffman)树,由发明它的人物命名,又称最优树,是一类带权路径最短的二叉树,主要用于数据压缩传输. 赫夫曼树的构造过程相对比较简单,要理解赫夫曼数,要先了解赫夫曼编码. 对一组出现频率不同 ...

  5. Python---哈夫曼树---Huffman Tree

    今天要讲的是天才哈夫曼的哈夫曼编码,这是树形数据结构的一个典型应用. !!!敲黑板!!!哈夫曼树的构建以及编码方式将是我们的学习重点. 老方式,代码+解释,手把手教你Python完成哈夫曼编码的全过程 ...

  6. 数据结构-哈夫曼树(python实现)

    好,前面我们介绍了一般二叉树.完全二叉树.满二叉树,这篇文章呢,我们要介绍的是哈夫曼树. 哈夫曼树也叫最优二叉树,与哈夫曼树相关的概念还有哈夫曼编码,这两者其实是相同的.哈夫曼编码是哈夫曼在1952年 ...

  7. 高级数据结构---赫(哈)夫曼树及java代码实现

    我们经常会用到文件压缩,压缩之后文件会变小,便于传输,使用的时候又将其解压出来.为什么压缩之后会变小,而且压缩和解压也不会出错.赫夫曼编码和赫夫曼树了解一下. 赫夫曼树: 它是一种的叶子结点带有权重的 ...

  8. Java实现WUST 1002: 哈夫曼树

    [问题描述] 根据给定的若干权值可以构造出一颗哈夫曼树.构造的哈夫曼树可能不唯一,但是按照下面的选取原则所构造出来的哈夫曼树应该是唯一的. (1)每次选取优先级最低的两个结点,优先级最低的作为左子树, ...

  9. C++ 漫谈哈夫曼树

    1. 前言 什么是哈夫曼树? 把权值不同的n个结点构造成一棵二叉树,如果此树满足以下几个条件: 此 n 个结点为二叉树的叶结点 . 权值较大的结点离根结点较近,权值较小的结点离根结点较远. 该树的带权 ...

随机推荐

  1. 如何简单使用tensorboard展示(一)

    我使用tensorboard中的graph做了展示,至于其它功能可以类推,其代码如下: import numpy as npimport tensorflow as tf x_img = np.arr ...

  2. File类---Day28

    概述 java.io.File 类是文件和目录路径名的抽象表示,主要用于文件和目录的创建.查找和删除等操作. 构造方法 1.public File(String pathname) :通过将给定的路径 ...

  3. open live writer安装以及代码高亮、折叠插件安装

    一.目的 方便在本地写博客,不用在浏览器上写. 二.open live writer的安装 下载open live writer 这是我的 链接:https://pan.baidu.com/s/1u8 ...

  4. LIBRARY_PATH和LD_LIBRARY_PATH

    LIBRARY_PATH是编译时指定的路径. LD_LIBRARY_PATH是运行时指定的动态链接库所在目录. 在运行一个可执行文件之前,可以通过ldd a.exe命令查看a.exe所依赖的动态链接库 ...

  5. JS--插件: 树Tree 开发与实现

    日常在Web项目开发时,经常会碰到树形架构数据的显示,从数据库中获取数据,并且显示成树形.为了方便,我们可以写一个javascript的一个跨浏览器树控件,后续可以重复使用.本节分享一个自己开发的JS ...

  6. Extjs 树菜单的自动展开数据的请求

    今天在做extjs开发的时候,在树菜单上遇到了一个坑,也许是我刚接触extjs 不熟的缘故 问题描述:后台设置的树自动展开,但是在前端总是只显示一条数据,但是数据确实都请求到了. 经过几个小时不屑的努 ...

  7. 爬取 豆瓣电影Top250

    目标 学习爬虫,爬豆瓣榜单,获取爬取静态页面信息的能力 豆瓣电影 Top 250  https://movie.douban.com/top250 代码 import requests from bs ...

  8. OEL7.6设置光盘YUM源

    先把原来的yum源改名让他失效 mv public-yum-ol7.repo public-yum-ol7.repo.bak 然后新建个yum源 [root@localhost yum.repos.d ...

  9. "轻"量级 Java Web 服务框架漫谈

    博文太长了, 还是先说下概要: 框架"轻量"与否可以从两方面来看待: 1) 框架本身的体量 - 例如小 jar 无依赖的苗条框架; 2) 用户使用框架是否获得各种便利而无阻隔(&q ...

  10. shell脚本遇到问题"$'\r': command not found"

    shell脚本写得一切正常,但是一执行就报错: line: XXX "$'\r': command not found" 问题原因:文件格式问题(虽然在window和linux上选 ...