C语言实现哈夫曼编码_哈夫曼编码压缩文件c语言

C语言实现哈夫曼编码_哈夫曼编码压缩文件c语言////霍夫曼编码//#include<stdio.h>#include<stdlib.h>#include<string.h>/**思路:用一个有序链表(从大到小)来保存节点,然后通过链表来构造霍夫曼树,再由霍夫曼树得到霍夫曼编码**/typedefstructhuffman_tree_node{intwe…………

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全家桶1年46,售后保障稳定

//
// 霍夫曼编码
//
 
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
 
/**思路:用一个有序链表(从大到小)来保存节点,然后通过链表来构造霍夫曼树, 再由霍夫曼树得到霍夫曼编码**/
 
typedef struct huffman_tree_node{
    int weight;//权重
    char c;//字符 非叶子节点为0
    struct huffman_tree_node * nextHuffmanTreeNode;//链表下一个节点
    struct huffman_tree_node * leftHuffmanTreeNode;//左节点
    struct huffman_tree_node * rightHuffmanTreeNode;//右节点
}HuffmanTreeNode; //霍夫曼树节点
 
typedef struct huffman_code{
    char *s;//编码 如  010, 00, ....
    int len;//编码长度
    char c;//字符
}HuffmanCode; //霍夫曼编码(可以用来保存结果)
 
/**
 * 创建一个节点
 * @param c  字符
 * @param weight 权重
 * @return
 */
HuffmanTreeNode * createHuffmanTreeNode(char c, int weight){
    HuffmanTreeNode * node = (HuffmanTreeNode *)calloc(1, sizeof(HuffmanTreeNode));
    node->c = c;
    node->weight = weight;
    node->nextHuffmanTreeNode = NULL;
    node->leftHuffmanTreeNode = NULL;
    node->rightHuffmanTreeNode = NULL;
    return node;
}
 
/**
 * [insert 插入节点到有序链表中]
 * @param  h [头节点指针]
 * @param  s [要插入的节点]
 * @return   [头节点]
 */
static HuffmanTreeNode * insert(HuffmanTreeNode * h, HuffmanTreeNode * s){
    if(h == NULL){ //插入第一个节点时 没有头节点
        return s;//s作为头节点
    }
    if(s->weight > h->weight){
        s->nextHuffmanTreeNode = h; //s作为头节点
        return s;
    }
 
    HuffmanTreeNode * tempHuffmanTreeNode = h; //中间变量 用于遍历
    HuffmanTreeNode * preHuffmanTreeNode = tempHuffmanTreeNode;//中间变量的前一个节点
    while(tempHuffmanTreeNode != NULL){
        if(s->weight < tempHuffmanTreeNode->weight){ //要插入的节点的学号比当前节点小
            preHuffmanTreeNode = tempHuffmanTreeNode;
            tempHuffmanTreeNode = tempHuffmanTreeNode->nextHuffmanTreeNode;
            continue;
        }
        //插入到中间
        preHuffmanTreeNode->nextHuffmanTreeNode = s;
        s->nextHuffmanTreeNode = tempHuffmanTreeNode;
        break;
    }
    if(tempHuffmanTreeNode == NULL){ //插入的节点比已有的节点都小
        preHuffmanTreeNode->nextHuffmanTreeNode = s;
    }
    return h;
}
 
/**
 * 移除最后一个节点(最小的那个) 并返回
 * @param node
 * @return
 */
HuffmanTreeNode * removeLastHuffmanTreeNode(HuffmanTreeNode * h){
    HuffmanTreeNode * tempHuffmanTreeNode = h; //中间变量 用于遍历
    HuffmanTreeNode * preHuffmanTreeNode = tempHuffmanTreeNode;//中间变量的前一个节点
    while(tempHuffmanTreeNode->nextHuffmanTreeNode != NULL){
            preHuffmanTreeNode = tempHuffmanTreeNode;
            tempHuffmanTreeNode = tempHuffmanTreeNode->nextHuffmanTreeNode;
        }
        preHuffmanTreeNode->nextHuffmanTreeNode = NULL;
    return tempHuffmanTreeNode;
}
 
/**
 * 链表转霍夫曼树
 * @param head
 * @return
 */
HuffmanTreeNode * createTree(HuffmanTreeNode * head){
    if(head == NULL){
        return NULL;
    }
    while (1){
        //获取最小的两个节点
        HuffmanTreeNode * node1 = removeLastHuffmanTreeNode(head);
        if(node1 == head){ //只有一个节点了 完成
            return node1;
        }
        HuffmanTreeNode * node2 = removeLastHuffmanTreeNode(head);
        if(node2 == head){ //最后一个节点
            head = NULL; //头节点置为NULL
        }
        //创建一个新的节点
        HuffmanTreeNode * newNode = createHuffmanTreeNode(0, node1->weight+node2->weight);
        //设置左节点
        newNode->leftHuffmanTreeNode = node1->weight <= node2->weight ? node1 : node2;
        //设置右节点
        newNode->rightHuffmanTreeNode = node1->weight <= node2->weight ? node2 : node1;
        //新节点插入到链表中,再次循环 直到链表中只有一个节点
        head = insert(head, newNode);
    }
}
 
/**
 * 字符串拷贝
 * @param s
 * @param len
 * @param dest
 */
void str_copy(char *s,int len ,char *dest){
    for(int i = 0; i < len; i++){
        dest[i] = s[i];
    }
}
 
/**
 * 霍夫曼编码
 * @param node 节点
 * @param s 编码的字符串 如 001,00,01...
 * @param len 编码字符串的长度
 */
void showCode(HuffmanTreeNode * node, char *s, int len){
    if(node->c != 0){ //到叶子节点了
        //打印编码结果(或保存到结构体中):
        printf("%c->%s\n", node->c, s);
        free(s);
        return;
    }
    //遍历左节点 编码增加一个0
    char * leftS = (char *)calloc(len+1, sizeof(char));
    str_copy(s, len, leftS);
    leftS[len] = '0';
    showCode(node->leftHuffmanTreeNode, leftS, len+1);
 
    //遍历右节点 编码增加一个1
    char * rightS = (char *)calloc(len+1, sizeof(char));
    str_copy(s, len, rightS);
    rightS[len] = '1';
    showCode(node->rightHuffmanTreeNode, rightS, len+1);
	
    free(s);
 
}
int main(){
    //创建节点
    HuffmanTreeNode * head = NULL;
    HuffmanTreeNode * node_a = createHuffmanTreeNode('A', 5);
    HuffmanTreeNode * node_b = createHuffmanTreeNode('B', 4);
    HuffmanTreeNode * node_c = createHuffmanTreeNode('C', 3);
    HuffmanTreeNode * node_d = createHuffmanTreeNode('D', 2);
    HuffmanTreeNode * node_e = createHuffmanTreeNode('E', 1);
    //插入到有序链表中
    head = insert(head,node_a);
    head = insert(head,node_b);
    head = insert(head,node_c);
    head = insert(head,node_d);
    head = insert(head,node_e);
    //转霍夫曼树
    HuffmanTreeNode *tree =  createTree(head);
    printf("huffman encode:\n");
    //获取编码
	char * s  = (char*)malloc(0);
    showCode(tree, s, 0);
 
}

Jetbrains全家桶1年46,售后保障稳定

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/209949.html原文链接:https://javaforall.net

(0)
上一篇 2025年7月30日 下午10:01
下一篇 2025年7月30日 下午10:43


相关推荐

  • OpenClaw龙虾实用指南 (安装包+教程)

    OpenClaw龙虾实用指南 (安装包+教程)OpenClaw 龙虾实用指南 安装包 教程

    2026年3月13日
    5
  • k8s–证书签发

    k8s–证书签发1.准备签发证书环境运维主机hdss-1-200.host.com上:2.安装CFSSL证书签发工具CFSSL:R1.2cfssl下载地址https://pkg.cfssl.org/R1.2/cfssl_linux-amd64cfssl-json下载地址https://pkg.cfssl.org/R1.2/cfssljson_linux-amd64cfssl-certinfo下载地址https://pkg.cfssl.org/R1.2/cfssl-certinfo_li…

    2022年5月9日
    96
  • pytest的使用_java中方法的调用或使用的地方

    pytest的使用_java中方法的调用或使用的地方Pytest执行用例规则Pytest在命令行中支持多种方式来运行和选择测试用例1.对某个目录下所有的用例pytest2.对模块中进行测试pytesttest_mod.py3.对文件夹进行

    2022年7月29日
    8
  • 第一个Java程序示例——Hello World!「建议收藏」

    第一个Java程序示例——Hello World!「建议收藏」来自http://www.weixueyuan.net/view/5947.html通过Eclipse运行程序启动Eclipse,在菜单中选择“文件–>新建–>Java项目”,弹出对话框:图1 新建项目填入项目名称,点击“完成”,创建项目成功,可以看到在 E:\java\workspace目录下多出了一个Demo文件夹。在菜单中选择“文

    2022年5月28日
    46
  • ESP8266开发之旅 基础篇② 如何安装ESP8266的Arduino开发环境

    ESP8266开发之旅 基础篇② 如何安装ESP8266的Arduino开发环境俗话说 工欲善其事必先利其器 基于 Arduino 平台开发 ESP8266 我们至少要分别准备好硬件和软件开发环境 1 硬件 ESP8266 模块 或者 ESP8266 开发板 笔者建议初学者直接使用开发板 2 软件 ArduinoIDE 开发环境 1 硬件 硬件环境主要分为两种 1 ESP8266 开发板 USB 线 笔者建议

    2026年3月18日
    1
  • 虚拟机连不上网所有可能我都遇到

    虚拟机连不上网所有可能我都遇到ping不通,主机访问不了虚拟机等等等等虚拟机ping不通VWnet8的ip、ping不通网关、ping不通百度、yum出现tryothermirror。虚拟机ping不通VWnet8的ip、ping不通网关、ping不通百度、yum出现tryothermirror。首先要编辑虚拟机网络为NAT模式,并设定网关、子网掩码。然后进入NAT设置里,将允许任何组织唯一标识符勾选,这个有时候系统默认没钩。然后进入DHCP设计起始ip和结束ip。进入本地网络,修改VWnet8适配器右键属性

    2022年6月26日
    32

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号