利用C语言编写完整代码。具体要求为: 从键盘任意输入一个由英文字母带空格构成的文本字符串,例如“I like Learning datastructure course”,首先根据该字符串创建Huffman树,之后进行Huffman编码,然后把该字符串的Huffman编码结果向屏幕输出。能够利用前述的Huffman编码结果对某0-1码流进行解码得到字符串,并做详细注释

时间: 2024-03-07 17:48:04 浏览: 17
好的,根据您的要求,我为您编写了以下代码,可以实现从键盘任意输入一个由英文字母带空格构成的文本字符串,利用 Huffman 编码和解码算法进行编码和解码,并将编码结果输出到屏幕上。 Huffman编码的过程是:首先统计每个字符在输入字符串中出现的频率,然后根据字符频率构建哈夫曼树,接着对哈夫曼树进行遍历,对于每个字符,输出其对应的二进制编码。解码过程是根据哈夫曼树以及编码结果进行逆向遍历,得到原始字符串。 代码如下: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> // 哈夫曼树节点结构体 typedef struct HuffmanNode { char data; // 节点保存的字符 int freq; // 字符出现的频率 struct HuffmanNode *left, *right; // 左右子节点 } HuffmanNode; // 哈夫曼编码结构体 typedef struct HuffmanCode { char data; // 字符 char *code; // 对应的哈夫曼编码 } HuffmanCode; // 统计字符串中每个字符出现的频率 int *countFreq(char *str) { int *freq = (int *)calloc(256, sizeof(int)); // 初始化为0 char *p = str; while (*p != '\0') { freq[*p++]++; } return freq; } // 创建哈夫曼树 HuffmanNode *createHuffmanTree(char *str) { // 统计字符串中每个字符出现的频率 int *freq = countFreq(str); // 创建哈夫曼节点数组 HuffmanNode **nodes = (HuffmanNode **)malloc(sizeof(HuffmanNode *) * 256); int count = 0; for (int i = 0; i < 256; i++) { if (freq[i] > 0) { HuffmanNode *node = (HuffmanNode *)malloc(sizeof(HuffmanNode)); node->data = (char)i; node->freq = freq[i]; node->left = node->right = NULL; nodes[count++] = node; } } free(freq); // 释放频率统计数组 // 构建哈夫曼树 while (count > 1) { // 找到频率最小的两个节点 int min1 = 0, min2 = 1; if (nodes[min1]->freq > nodes[min2]->freq) { int temp = min1; min1 = min2; min2 = temp; } for (int i = 2; i < count; i++) { if (nodes[i]->freq < nodes[min1]->freq) { min2 = min1; min1 = i; } else if (nodes[i]->freq < nodes[min2]->freq) { min2 = i; } } // 创建新节点 HuffmanNode *newNode = (HuffmanNode *)malloc(sizeof(HuffmanNode)); newNode->data = '\0'; newNode->freq = nodes[min1]->freq + nodes[min2]->freq; newNode->left = nodes[min1]; newNode->right = nodes[min2]; // 将新节点加入节点数组中 nodes[min1] = newNode; nodes[min2] = nodes[count - 1]; count--; } // 返回哈夫曼树的根节点 return nodes[0]; } // 释放哈夫曼树的内存 void freeHuffmanTree(HuffmanNode *root) { if (root != NULL) { freeHuffmanTree(root->left); freeHuffmanTree(root->right); free(root); } } // 递归遍历哈夫曼树,生成哈夫曼编码 void generateHuffmanCode(HuffmanNode *root, char *code, int depth, HuffmanCode *huffmanCodes) { if (root->left == NULL && root->right == NULL) { // 生成哈夫曼编码 code[depth] = '\0'; int i; for (i = 0; huffmanCodes[i].data != '\0'; i++) { if (huffmanCodes[i].data == root->data) { huffmanCodes[i].code = (char *)malloc(sizeof(char) * (depth + 1)); strcpy(huffmanCodes[i].code, code); break; } } if (huffmanCodes[i].data == '\0') { huffmanCodes[i].data = root->data; huffmanCodes[i].code = (char *)malloc(sizeof(char) * (depth + 1)); strcpy(huffmanCodes[i].code, code); huffmanCodes[i + 1].data = '\0'; } return; } // 递归遍历左右子树 code[depth] = '0'; generateHuffmanCode(root->left, code, depth + 1, huffmanCodes); code[depth] = '1'; generateHuffmanCode(root->right, code, depth + 1, huffmanCodes); } // 利用哈夫曼编码将字符串进行编码 char *encodeString(char *str, HuffmanCode *huffmanCodes) { // 计算编码后字符串的长度 int len = 0; char *p = str; while (*p != '\0') { for (int i = 0; huffmanCodes[i].data != '\0'; i++) { if (*p == huffmanCodes[i].data) { len += strlen(huffmanCodes[i].code); break; } } p++; } // 分配空间并进行编码 char *codeStr = (char *)malloc(sizeof(char) * (len + 1)); char *q = codeStr; p = str; while (*p != '\0') { for (int i = 0; huffmanCodes[i].data != '\0'; i++) { if (*p == huffmanCodes[i].data) { strcat(q, huffmanCodes[i].code); q += strlen(huffmanCodes[i].code); break; } } p++; } *q = '\0'; return codeStr; } // 利用哈夫曼编码将0-1编码流进行解码 char *decodeString(char *codeStr, HuffmanNode *root) { int len = strlen(codeStr); char *str = (char *)malloc(sizeof(char) * (len + 1)); HuffmanNode *p = root; char *q = str; for (int i = 0; i < len; i++) { if (codeStr[i] == '0') { p = p->left; } else { p = p->right; } if (p->left == NULL && p->right == NULL) { *q++ = p->data; p = root; } } *q = '\0'; return str; } int main() { char str[1000]; printf("请输入字符串:"); gets(str); // 创建哈夫曼树 HuffmanNode *root = createHuffmanTree(str); // 生成哈夫曼编码 char code[256]; HuffmanCode huffmanCodes[256]; memset(huffmanCodes, 0, sizeof(huffmanCodes)); generateHuffmanCode(root, code, 0, huffmanCodes); // 输出哈夫曼编码表 printf("字符\t频率\t编码\n"); for (int i = 0; huffmanCodes[i].data != '\0'; i++) { printf("%c\t%d\t%s\n", huffmanCodes[i].data, countFreq(str)[huffmanCodes[i].data], huffmanCodes[i].code); } // 编码 char *codeStr = encodeString(str, huffmanCodes); printf("编码结果:%s\n", codeStr); // 解码 char *decodeStr = decodeString(codeStr, root); printf("解码结果:%s\n", decodeStr); // 释放内存 freeHuffmanTree(root); free(codeStr); free(decodeStr); for (int i = 0; huffmanCodes[i].data != '\0'; i++) { free(huffmanCodes[i].code); } return 0; } ``` > 注意:为了简化代码,没有对输入的字符串进行长度限制。实际应用中需要根据具体情况进行限制。

相关推荐

最新推荐

recommend-type

peak-linux-driver-8.15.2.tar

peak-linux-driver-8.15.2.tar
recommend-type

VSCodeUserSetup-x64-1.86.1.exe

VSCodeUserSetup-x64-1.86.1
recommend-type

毕业设计使用ncnn在ios+android上部署yolov5源码+详细说明.zip

高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip 高分毕业设计 毕业设计源码 使用ncnn在ios+android上部署yolov5源码+详细说明.zip
recommend-type

课设毕设基于SSM的医院远程诊断系统-LW+PPT+源码可运行.zip

课设毕设基于SSM的医院远程诊断系统--LW+PPT+源码可运行.
recommend-type

node-v0.10.25-sunos-x86.tar.gz

Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

实现实时数据湖架构:Kafka与Hive集成

![实现实时数据湖架构:Kafka与Hive集成](https://img-blog.csdnimg.cn/img_convert/10eb2e6972b3b6086286fc64c0b3ee41.jpeg) # 1. 实时数据湖架构概述** 实时数据湖是一种现代数据管理架构,它允许企业以低延迟的方式收集、存储和处理大量数据。与传统数据仓库不同,实时数据湖不依赖于预先定义的模式,而是采用灵活的架构,可以处理各种数据类型和格式。这种架构为企业提供了以下优势: - **实时洞察:**实时数据湖允许企业访问最新的数据,从而做出更明智的决策。 - **数据民主化:**实时数据湖使各种利益相关者都可
recommend-type

SPDK_NVMF_DISCOVERY_NQN是什么 有什么作用

SPDK_NVMF_DISCOVERY_NQN 是 SPDK (Storage Performance Development Kit) 中用于查询 NVMf (Non-Volatile Memory express over Fabrics) 存储设备名称的协议。NVMf 是一种基于网络的存储协议,可用于连接远程非易失性内存存储器。 SPDK_NVMF_DISCOVERY_NQN 的作用是让存储应用程序能够通过 SPDK 查询 NVMf 存储设备的名称,以便能够访问这些存储设备。通过查询 NVMf 存储设备名称,存储应用程序可以获取必要的信息,例如存储设备的IP地址、端口号、名称等,以便能
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。