当前位置: 首页 > 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"...

计算机竞赛 基于CNN实现谣言检测 - python 深度学习 机器学习

文章目录 1 前言1.1 背景 2 数据集3 实现过程4 CNN网络实现5 模型训练部分6 模型评估7 预测结果8 最后 1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于CNN实现谣言检测 该项目较为新颖&#xff0c;适合作为竞赛课题方向&#xff0c;学长非常推荐&am…...

框架(Git基础详解及Git在idea中集成步骤)

目录 基础&#xff1a; idea集成Git并添加项目到git仓库 1.idea集成git&#xff0c;集成.git.exe文件 2.初始化本地Git仓库项目 3. 将工作区代码添加到暂存区 4.将暂存区代码添加到本地仓库 5.Git本地库操作 Idea集成Gitee并提交代码到第三方库 1.setting里搜索gitee 2.添…...

0基础学习VR全景平台篇 第88篇:智慧眼-成员管理

一、功能说明 成员管理&#xff0c;是指管理智慧眼项目的成员&#xff0c;拥有相关权限的人可以进行添加成员、分配成员角色、设置成员分类、修改成员以及删除成员五项操作。但是仅限于管理自己的下级成员&#xff0c;上级成员无权管理。 二、前台操作页面 登录智慧眼后台操…...

DSO 系列文章(2)——DSO点帧管理策略

文章目录 1.点所构成的残差Residual的管理1.1.前端残差的状态1.2.后端点的残差的状态1.3.点的某个残差的删除 2.点Point的管理2.1.如何删除点——点Point的删除2.2.边缘化时删除哪些点&#xff1f; 3.帧FrameHessian的管理 DSO代码注释&#xff1a;https://github.com/Cc19245/…...

无需公网IP——搭建web站点

文章目录 概述使用 Raspberry Pi Imager 安装 Raspberry Pi OS设置 Apache Web 服务器测试 web 站点安装静态样例站点将web站点发布到公网安装 Cpolar内网穿透cpolar进行token认证生成cpolar随机域名网址生成cpolar二级子域名将参数保存到cpolar配置文件中测试修改后配置文件配…...

swift 项目集成友盟推送

1, 需要用桥接文件 , 不然引用不到依赖库 2, 可以用测试模式测试, 可以debug 3, 测试模式获取deviceToken, 添加测试设备 deviceToken获取方法 func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) { le…...

Unity之用Transform 数组加多个空物体-->简单地控制物体按照指定路线自动行驶

文章目录 **原理解释**&#xff1a;**带注释的代码**&#xff1a;实际运用 当你需要实现物体按照指定路线行驶时&#xff0c;你可以通过以下步骤来实现&#xff1a; 原理解释&#xff1a; 路径点&#xff1a;你需要定义一系列路径点&#xff0c;这些点将构成物体行驶的路线。每…...

交换机生成树STP

生成树协议&#xff08;spanning-tree-protocol,stp&#xff09;&#xff1a;在具有物理环路的交换机网络上生成没有回路的逻辑网络的方法&#xff0c;生成树协议使用生成树算法&#xff0c;在一个具有冗余路径的容错网络中计算出一个无环路的路径&#xff0c;使一部分端口处于…...

3.微服务概述

1.大型网络架构变迁 SOA与微服务最大的差别就是服务拆分的细度&#xff0c;目前大多数微服务实际上是SOA架构&#xff0c;真正的微服务应该是一个接口对应一个服务器&#xff0c;开发速度快、成本高&#xff1b; 微服务SOA能拆分的就拆分是整体的&#xff0c;服务能放一起的都…...

cloud_mall-notes02

1、多条件分页查询page ApiOperation("多条件分页查询xxxx")GetMapping("page")PreAuthorize("hasAuthority(模块权限:权限:page)")public ResponseEntity<Page<实体类>> loadxxxxPage(Page<实体类> page,实体类 domain) {pag…...

石家庄自助建站模板/seo优化的网站

参考链接 RUNOOB.COM 多态处理 我们在前面已经讲过继承的定义&#xff0c;但是当我们遇到继承类和基类中的方法相同时&#xff0c;程序该如何运行呢&#xff1f;这时&#xff0c;就需要引入C中多态操作。多态调用时&#xff0c;会根据函数对象类型执行不同函数&#xff1a; …...

wordpress中文别名分类目录/seow

根据于渊老师的《Orange’s 一个操作系统的实现》我开始了操作系统的制作 首先进行了helloworld的编写&#xff0c;大体过程是&#xff1a; 1.了解什么语言编写操作系统 书中给出&#xff1a;汇编与c&#xff0c;但是helloworld编写仅仅是汇编&#xff0c;我也看不懂 直接复…...

html5手机网站开发工具/seo优化招聘

为什么80%的码农都做不了架构师&#xff1f;>>> 最近在开发中碰到当手机没有网络的时候&#xff0c;WebView加载本地缓存出了问题&#xff0c;界面变得很乱&#xff0c;初步断定是样式表没有加载上来。 WebView的缓存策略是这样的&#xff1a;webSettings.setCache…...

潍坊网站建设 潍坊做网站/百度访问量统计

接着昨天的继续谈关于微信新出的这个js框架&#xff0c;今天主要谈一个页面的创建到布局的详细步骤。 一.创建一个完整页面 页面你可以创建在项目的任何节点&#xff0c;只要你在入口文件正确引入创建该页面的路径就可使用。 上面使用红色矩形包含的目录&#xff0c;是我新增的…...

衡阳市建设协会网站/100个免费推广网站

在SQL Server 2005/2008中的当前数据库中遍历所有表显示所有表的行数 SQL Server DECLARE CountTableRecords CURSOR READ_ONLY FOR SELECT sst.name, Schema_name(sst.schema_id) FROM sys.tables sst WHERE sst.TYPE U DECLARE name VARCHAR(80), sche在SQL Server 2005/200…...

博客wordpress主题/软件开发需要多少资金

方案介绍亚马逊云科技 WAF 是一个网页应用的防火墙服务。它能帮助保护您的网页应用或网页 API 应对常见的网页攻击。在 WAF 部署小指南(一) 中&#xff0c;我们讨论了 WAF 的原理&#xff0c;如何使用 WAF&#xff0c;以及如何将 WAF 日志存储在 Amazon S3 中&#xff0c;并通过…...