当前位置: 首页 > news >正文

数据结构作业——哈夫曼树

/*【基本要求】
(1) 从文件中读出一篇英文文章,包含字母和空格等字符。
(2) 统计各个字符出现的频度。
(3) 根据出现的频度,为每个出现的字符建立一个哈夫曼编码,并输出。
(4)  输入一个字符串,为其编码并输出。
(5) 输入一串编码,为其译码并输出*//*【演示结果】
(1)显示英文文章及各字符出现的频率。
(2)显示每个字符的哈夫曼编码。
(3)文件读入一文本,显示对其编码结果,并存盘
(4)文件读入一组编码,显示对其译码结果,并存盘*/#include<stdio.h>
using namespace std;#define N 128//最多128个字符种类//数据存储结构
typedef struct{char data;//数据int weight;//数据权重int lchild,rchild,parent;//左右结点及双亲
}HumfNode;//词频统计存储结构
typedef struct{char data;int freq;
}Datafreq;//编码存储结构
typedef struct{int bits[128];存放编码0、1的数组int start;
}HCode;

代码:

#include<stdio.h>
#include<string.h>
#include<conio.h>
#include<stdlib.h>
#include<fstream>
#include<iostream>
using namespace std;
#define N 128                                            //最大叶子结点数typedef struct
{char data;                                          //编码对应的字符int weight;                                         //结点的权值int lchild, rchild, parent;
}HumfNode;typedef struct
{char data;int freq;
}Datafreq;typedef struct
{int bits[128];                                        //存放哈夫曼编码的字符数组int start;                                           //编码的起始位置
}HCode;void FreqNode(char* st, Datafreq str[])                   //统计单词和空格与其频率
{int i, j, k, num[128];char* p = st;for (i = 0; i < 128; i++){str[i].freq = 0;}//初始化频度结点的频度for (i = 0; i < 128; i++)num[i] = 0;//printf("英文文章如下:");while (*p != NULL){num[int(*p)]++;p++;}j = 0;for (i = 0; i < 128; i++){str[i].data = char(i);str[i].freq = num[i];//统计每个结点的权重和内容}printf("\n");printf("频度如下(ascll码由小到大排列):");for (i = 0; i < 128; i++){if (str[i].freq != '\0'){cout << str[i].data << str[i].freq << " ";}}printf("\n");}//功能实现的是统计文档中的各个字符的出现频率
//将整个ascll码表全部存储,并将其频度(权重)和内容放入哈夫曼结点中
void CreatHufmTree(HumfNode tree[], Datafreq str[], int n)                    //建立哈夫曼树                 
{int m1, m2, i, l, r, k;Datafreq* p = str;for (i = 0; i < 2 * n - 1; i++){tree[i].lchild = tree[i].rchild = tree[i].parent = -1;tree[i].weight = 0;}for (i = 0; i < n; i++){tree[i].data = p[i].data;tree[i].weight = p[i].freq;}for (i = n; i < 2 * n - 1; i++){m1 = m2 = 32767;l = r = -1;for (k = 0; k < i; k++){if (tree[k].parent == -1 && tree[k].weight <= m1){m2 = m1;r = l;m1 = tree[k].weight;l = k;}else if (tree[k].parent == -1 && tree[k].weight <= m2){m2 = tree[k].weight;r = k;}else{}}tree[i].weight = tree[l].weight + tree[r].weight;tree[i].lchild = l;tree[i].rchild = r;tree[l].parent = i;tree[r].parent = i;if (tree[i].weight == tree[l].weight){tree[i].data = tree[l].data;tree[i].weight = tree[l].weight;tree[i].lchild = tree[i].rchild = -1;}else if (tree[i].weight == tree[r].weight){tree[i].data = tree[r].data;tree[i].weight = tree[r].weight;tree[i].lchild = tree[i].rchild = -1;}//下标为i的新结点成为权值最小的两个结点双亲//新结点的权值为两个结点权值之和//权值最小的结点是新结点的左孩子//权值次最小的结点为右孩子}}//建立哈夫曼树void HufmCode(HumfNode tree[], HCode hcd[], int n)                    //哈夫曼编码的生成
{int i, f, c, k;HCode cd;                                          //用于临时存放编码串for (i = 0; i < 128; i++){for (int r = 0; r < N; r++){cd.bits[r] = 2;}cd.start = n - 1;c = i;                                                    //从叶子结点开始往上回溯f = tree[i].parent;//找到它的双亲结点while (f != -1)                                             //回溯到根结点{//&& tree[f].weight != tree[c].weightif (tree[f].lchild == c && tree[tree[f].rchild].weight != tree[f].weight && tree[tree[f].lchild].weight != tree[f].weight){cd.bits[cd.start] = 0;cd.start--;c = f;f = tree[c].parent;}else if (tree[f].rchild == c && tree[tree[f].rchild].weight != tree[f].weight && tree[tree[f].lchild].weight != tree[f].weight){cd.bits[cd.start] = 1;cd.start--;c = f;f = tree[c].parent;}else{tree[f].data = tree[tree[f].rchild].data;tree[f].weight = tree[tree[f].rchild].weight;tree[f].lchild = tree[f].rchild = -1;c = f;f = tree[c].parent;}}//cd.start++;hcd[i] = cd;}printf("输出哈夫曼编码:\n");for (i = 0; i < n; i++){if (tree[i].weight != 0){printf("%c\n", tree[i].data);for (k = hcd[i].start + 1; k < n; k++){printf("%d", hcd[i].bits[k]);}printf("\n");}}
}string TsCode( char a[], HumfNode tree[], int n)                          //哈夫曼树的译码
{char* p = a;int i = 0;int k = 0;i = 2 * n - 2;string tsresult;//将树根结点的下标赋i,从根结点出发向下搜索a = p;//unsigned long len = strlen(a);printf("译码结果如下:");while (*a != '2'&&*a!='\0'){if (*a == '0'){//printf("%d\n", tree[i].weight);i = tree[i].lchild;if ((tree[i].lchild == -1) && (tree[i].rchild == -1)){//printf("%d\n", tree[i].weight);printf("%c", tree[i].data);tsresult+=tree[i].data;i = 2 * n - 2;k++;}a++;}else if (*a == '1'){//printf("%d\n", tree[i].weight);i = tree[i].rchild;if ((tree[i].lchild == -1) && (tree[i].rchild == -1)){//printf("%d\n", tree[i].weight);printf("%c", tree[i].data);tsresult += tree[i].data;i = 2 * n - 2;k++;}a++;}}return tsresult;
}void outputfiles(string file,string a)
{ofstream fout(file);fout << a;fout.close();}void outputfile(string file, HCode hcd[], HumfNode tree[], int n)
{int i, k;ofstream fout(file);if (!fout){cout << "文件不能打开" << endl;}else{// 输出到磁盘文件for (i = 0; i < n; i++){if (tree[i].data != '\0' && tree[i].weight != 0){fout << tree[i].data << ":";for (k = hcd[i].start + 1; k < n; k++)if (hcd[i].bits[k] != 2){fout << hcd[i].bits[k];}fout << endl;printf("\n");}}//关闭文件输出流fout.close();}
}char* openfile(string file, char* st)                            //打开并显示文件
{char ch;int i = 0;ifstream infile;infile.open(file.data());   //将文件流对象与文件连接起来 while (!infile.eof()){infile.get(ch); //get( )函数从相应的流文件中读出一个字符,并将其返回给变量chif (infile.fail()){break;}st[i] = ch;//cout << ch;i++;}infile.close();       //关闭文件return st;
}void Getcode(char* bit, HumfNode tree[], HCode hcd[], int n)
{char* p = bit;int i = 0, k;while (*(bit + i) != '\0'){for (k = 0; k < n; k++){if (tree[k].data == *(bit + i)){for (int r = hcd[k].start; r < n; r++)printf("%d", hcd[k].bits[r]);}}i++;}//while (*p != '\0')//{//	int i = 1, k;//	while (i <= n)//	{//		if (tree[i].data == *p)//		{//			//	printf("输出哈夫曼编码:\n");//			//	printf("%c",tree[i].data);//			for (k = hcd[i].start; k <= n; k++)//				printf("%c", hcd[i].bits[k]);//			//	printf("\n");//		}//		i++;//		//	else//		//		i++;//	}//	p++;//}
}void main()
{int i, j, k, t = 0, m, b;char x;int n = 128;Datafreq str[128], stt[128], sft[128], num[128];char st[1000], bm[200], sd[50], sf[50], sm[50];HumfNode tree[2 * N - 1], st_tree[2 * N - 1], sf_tree[2 * N - 1];                                 //用于存放树中所有结点HCode hcd[N], st_hcd[N], hst[N];   //用于存放字符的哈夫曼编码char* ss, * yima;string tscode;while (1){printf("******************************************************************************\n");printf("******************************************************************************\n");printf("**   1.从文件中读出一篇英文文章,包含字母和空格等字符。                     **\n");printf("**   2.统计各个字符出现的频度,为每个出现的字符建立一个哈夫曼编码,并输出。 **\n");printf("**   3.输入一个字符串,为其编码并输出。                                     **\n");printf("**   4.输入一串编码,为其译码并输出。                                       **\n");printf("**   5.退出                                                                 **\n");printf("******************************************************************************\n");printf("******************************************************************************\n");scanf_s("%d", &x);switch (int(x)){case 1:for (int y = 0; y < 1000; y++){st[y] = '\0';}ss = openfile("D:\\mathess\\eee.txt", st);printf("英文文章如下:");while (*ss != '\0'){printf("%c", *ss);ss++;}printf("\n");break;case 2:  FreqNode(st, str);CreatHufmTree(tree, str, n);//cout << tree[65].data;HufmCode(tree, hcd, n);outputfile("D:\\mathess\\eeecode.txt", hcd, tree, n);break;case 3: printf("请输入一个字符串:");scanf_s("%s", &sd, 50);FreqNode(sd, stt);CreatHufmTree(st_tree, stt, n);HufmCode(st_tree, st_hcd, n);//Getcode(sd, tree, hcd, n);break;case 4:	printf("请输入一个字符串(为后面的译码内容提供编码参考):");scanf_s("%s", &sf, 50);FreqNode(sf, sft);CreatHufmTree(sf_tree, sft, n);HufmCode(sf_tree, hst, n);for (int i = 0; i < 200; i++){bm[i] = '2';}yima = openfile("D:\\mathess\\xuyaoyima.txt", bm);i = 0;printf("文档的一串编码为:");while (*yima != '\0'){if (*yima == '0' || *yima == '1'){printf("%c", *yima);yima++;}else{break;}}printf("\n");//scanf_s("%s", bm, 200);//printf("译码后的结果:");tscode=TsCode( bm, sf_tree, n);//printf("%s", tscode);//printf("%c", * Tscode);outputfiles("D:\\mathess\\tscode.txt", tscode);printf("\n");break;case 5: exit(0);//	default: printf("输入有误,请重新输入");}}
}

运行结果:

 

 

从文件读取英文文章,并显示读取后的文章内容

 

 

将其统计频率进行输出,并将编码结果存盘

 

输入需要编码的字符串,并将译码结果输出

 

输入需要译码的编码,进行译码,并将译码结果输出,存盘,经过判断结果正确。

 

 

相关文章:

数据结构作业——哈夫曼树

/*【基本要求】 &#xff08;1&#xff09; 从文件中读出一篇英文文章&#xff0c;包含字母和空格等字符。 &#xff08;2&#xff09; 统计各个字符出现的频度。 &#xff08;3&#xff09; 根据出现的频度&#xff0c;为每个出现的字符建立一个哈夫曼编码&#xff0c;并输出。…...

Python XML处理中级篇:深入探索lxml库

lxml库是Python中处理XML和HTML文档的强大库&#xff0c;提供了丰富的API以进行各种操作。在初级篇中&#xff0c;我们介绍了如何使用lxml库解析、访问和修改XML文档。在这篇中级篇中&#xff0c;我们将更深入地探讨如何使用lxml库&#xff0c;包括如何创建XML文档&#xff0c;…...

岩棉革新——洛科威推出NGF新一代岩棉产品

作为全球领先的岩棉制品生产商&#xff0c;洛科威公司基于在岩棉性能革新领域八十多年的深入研究和生产工艺的不断优化&#xff0c;在中国市场正式推出NGF新一代岩棉制品&#xff0c;并在上海国际绿色建筑建材博览会和2023国际绿色低碳技术展上正式发布。 洛科威NGF产品作为革…...

关于 docker 基础题目

1.安装docker服务&#xff0c;配置镜像加速器 http://t.csdn.cn/E3zQ8 2.下载系统镜像&#xff08;Ubuntu、 centos&#xff09; 执行该命令后&#xff0c;Docker会自动从Docker Hub镜像库中下载Ubuntu镜像&#xff0c;并将其保存到本地计算机上: [rootmaster ~]# docker pull …...

Skywalking全链路追踪【学习笔记】

Skywalking全链路追踪的服务搭建&#xff0c;使用docker进行安装。 搭建服务 搭建【ES】 # 拉取 docker pull docker.elastic.co/elasticsearch/elasticsearch:7.17.10 # 启动 docker run -p 127.0.0.1:9200:9200 -p 127.0.0.1:9300:9300 -e "discovery.typesingle-nod…...

Sphinx——Python生成API文档

1、简介 Sphinx是Python文档生成器&#xff0c;它基于reStructuredText标记语言&#xff0c;可自动根据项目生成HTML&#xff0c;PDF等格式的文档&#xff0c;无数著名项目的文档均用Sphinx生成&#xff0c;如机器学习库scikit-learn、交互式神器Jupyter Notebook sphinx是一…...

倒计时动效

1. 效果 2. html <div class"count"><span>3</span><span>2</span><span>1</span> </div>3. css body {width: 100vw;height: 100vh;overflow: hidden;display: flex;justify-content: center;align-items: cente…...

安卓主板定制_电磁屏/电容屏安卓平板基于MTK联发科方案定制

定制化行业平板 在各行各业中的地位越来越重要&#xff0c;甚至在行业转型和发展中发挥着不可替代的作用。随着工业化社会的快速发展&#xff0c;工业生产对智控设备要求越来越高&#xff0c;运用的范畴也越来越普遍广泛&#xff0c;工业级平板就是其中一种应用广泛的设备。 新…...

Unity 之 ScreenPointToRay() (将点转换成射线的方法)

文章目录 ScreenPointToRay() ScreenPointToRay() ScreenPointToRay() 是Unity中Camera类的一个方法&#xff0c;用于将屏幕上的一个点转换为一条射线。这条射线的起点是摄像机在屏幕上对应的点&#xff0c;方向是从摄像机出发指向那个点。这在进行射线命中检测时非常有用&…...

C++ 线程池

目录 一、线程池实现原理 二、定义线程池的结构 三、创建线程池实例 四、添加工作的线程的任务函数 五、管理者线程的任务函数 六、往线程池中添加任务 七、获取线程池工作的线程数量与活着的线程数量 八、线程池的销毁 一、线程池实现原理 线程池的组成主要分为3个部…...

测试框架pytest教程(6)钩子函数hook开发pytest插件

pytest hook 函数也叫钩子函数&#xff0c;pytest 提供了大量的钩子函数&#xff0c;可以在用例的不同生命周期自动调用。 比如&#xff0c;在测试用例收集阶段&#xff0c;可利用 hook 函数修改测试用例名称的编码。 pytest的hook是基于Python的插件系统实现的&#xff0c;使…...

【Rust】Rust学习 第十七章Rust 的面向对象特性

面向对象编程&#xff08;Object-Oriented Programming&#xff0c;OOP&#xff09;是一种模式化编程方式。对象&#xff08;Object&#xff09;来源于 20 世纪 60 年代的 Simula 编程语言。这些对象影响了 Alan Kay 的编程架构中对象之间的消息传递。他在 1967 年创造了 面向对…...

Redis系列(四):哨兵机制详解

首发博客地址 https://blog.zysicyj.top/ 前面我们说过&#xff0c;redis采用了读写分离的方式实现高可靠。后面我们说了&#xff0c;为了防止主节点压力过大&#xff0c;优化成了主-从-从模式 思考一个问题&#xff0c;主节点此时挂了怎么办 这里主从模式下涉及到的几个问题&a…...

一个滚动框高度动态计算解决方案

需求描述&#xff0c;一个嵌套了很多层div或者其他标签的内容框&#xff0c;而它的外层没有设置高度&#xff0c;或者使用百分比&#xff0c;而本容器需要设置高度来实现滚动&#xff0c;要么写死px高度&#xff0c;但是不能自适应&#xff0c;此时需要一个直系父容器&#xff…...

Android瀑布流

以下是一个简单的示例代码&#xff0c;演示如何在Android Studio中解析指定网页的图片URL&#xff0c;并展示在错乱瀑布流布局中&#xff1a; 1. 添加网络权限&#xff1a;在项目的AndroidManifest.xml文件中添加以下权限&#xff1a; <uses-permission android:name"…...

Ubuntu搭建CT_ICP里程计的环境暨CT-ICP部署

CT-ICP部署以及运行复现过程 0.下载资源&#xff0c;并按照github原网址的过程进行。1.查看所需要的各个部分的版本。2.安装clang编译器3.进行超级构建3.1标准进行3.2构建过程中遇到的问题 4.构建并安装CT-ICP库4.1标准进行4.2遇到的问题及解决办法 5.构建 CT-ICP 的 ROS 包装5…...

微信小程序全局事件订阅eventBus

微信小程序全局事件订阅 在Vue开发中&#xff0c;我们可能用过eventBus来解决全局范围内的事件订阅及触发逻辑&#xff0c;在微信小程序的开发中我们可能也也会遇到同样的需求&#xff0c;那么我们尝试下在小程序&#xff08;原生小程序开发&#xff09;中实现类似eventBus的事…...

华为云cce发布若依前后分离版:2.nginx镜像操作

下载nginx docker的官方镜像。 docker资源很难找,我在我的空间上传了一个,需要的话可以下载: https://download.csdn.net/download/axe6404/88225311 下载后,请用以下方法安装 2.1 导入docker 官方nginx镜像。 将镜像包nginx docker镜像包nginx-dockerimage.tar放…...

TCP协议报文结构

TCP是什么 TCP&#xff08;传输控制协议&#xff09;是一种面向连接的、可靠的、全双工的传输协议。它使用头部&#xff08;Header&#xff09;和数据&#xff08;Data&#xff09;来组织数据包&#xff0c;确保数据的可靠传输和按序传递。 TCP协议报文结构 下面详细阐述TCP…...

Day14-2-NodeJS后端开发流程

Day14-NodeJS后端工程化流程 一 apifox工具 apifox是目前最好的接口调试工具 1 环境搭建 安装登录创建项目接口里面创建对应文件夹在指定的文件夹里面创建接口2 GET请求 1 apifox发送GET请求 2 后端接收GET请求 router.get("/getUserinfo"...

MongoDB学习和应用(高效的非关系型数据库)

一丶 MongoDB简介 对于社交类软件的功能&#xff0c;我们需要对它的功能特点进行分析&#xff1a; 数据量会随着用户数增大而增大读多写少价值较低非好友看不到其动态信息地理位置的查询… 针对以上特点进行分析各大存储工具&#xff1a; mysql&#xff1a;关系型数据库&am…...

PL0语法,分析器实现!

简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...

人机融合智能 | “人智交互”跨学科新领域

本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...

(一)单例模式

一、前言 单例模式属于六大创建型模式,即在软件设计过程中,主要关注创建对象的结果,并不关心创建对象的过程及细节。创建型设计模式将类对象的实例化过程进行抽象化接口设计,从而隐藏了类对象的实例是如何被创建的,封装了软件系统使用的具体对象类型。 六大创建型模式包括…...

嵌入式学习之系统编程(九)OSI模型、TCP/IP模型、UDP协议网络相关编程(6.3)

目录 一、网络编程--OSI模型 二、网络编程--TCP/IP模型 三、网络接口 四、UDP网络相关编程及主要函数 ​编辑​编辑 UDP的特征 socke函数 bind函数 recvfrom函数&#xff08;接收函数&#xff09; sendto函数&#xff08;发送函数&#xff09; 五、网络编程之 UDP 用…...

起重机起升机构的安全装置有哪些?

起重机起升机构的安全装置是保障吊装作业安全的关键部件&#xff0c;主要用于防止超载、失控、断绳等危险情况。以下是常见的安全装置及其功能和原理&#xff1a; 一、超载保护装置&#xff08;核心安全装置&#xff09; 1. 起重量限制器 功能&#xff1a;实时监测起升载荷&a…...

[QMT量化交易小白入门]-六十二、ETF轮动中简单的评分算法如何获取历史年化收益32.7%

本专栏主要是介绍QMT的基础用法,常见函数,写策略的方法,也会分享一些量化交易的思路,大概会写100篇左右。 QMT的相关资料较少,在使用过程中不断的摸索,遇到了一些问题,记录下来和大家一起沟通,共同进步。 文章目录 相关阅读1. 策略概述2. 趋势评分模块3 代码解析4 木头…...

MeanFlow:何凯明新作,单步去噪图像生成新SOTA

1.简介 这篇文章介绍了一种名为MeanFlow的新型生成模型框架&#xff0c;旨在通过单步生成过程高效地将先验分布转换为数据分布。文章的核心创新在于引入了平均速度的概念&#xff0c;这一概念的引入使得模型能够通过单次函数评估完成从先验分布到数据分布的转换&#xff0c;显…...

Centos 7 服务器部署多网站

一、准备工作 安装 Apache bash sudo yum install httpd -y sudo systemctl start httpd sudo systemctl enable httpd创建网站目录 假设部署 2 个网站&#xff0c;目录结构如下&#xff1a; bash sudo mkdir -p /var/www/site1/html sudo mkdir -p /var/www/site2/html添加测试…...

如何在Spring Boot中使用注解动态切换实现

还在用冗长的if-else或switch语句管理多个服务实现? 相信不少Spring Boot开发者都遇到过这样的场景:需要根据不同条件动态选择不同的服务实现。 如果告诉你可以完全摆脱条件判断,让Spring自动选择合适的实现——只需要一个注解,你是否感兴趣? 本文将详细介绍这种优雅的…...