第4章 二叉树和BST

news2025/1/22 20:53:28

树与二叉树

1. 基本概念

树是一种非线性结构,其严格的数学定义是:如果一组数据中除了第一个节点(第一个节点称为根节 点,没有直接前驱节点)之外,其余任意节点有且仅有一个直接前驱,有零个或多个直接后继,这样 的一组数据形成一棵树。这种特性简称为一对多的逻辑关系。即用于描述具有层次关系,类似组织架 构关系的一种数据结构。

树的组成:根,分支,叶子

2. 常见例子

日常生活中,很多数据的组织形式本质上是一棵树。比如一个公司中的职员层级关系,一个学校中的 院系层级关系,淘汰赛中的各次比赛队伍,一个家族中的族谱成员关系等,这些都是树状逻辑结构。 由于树状结构表现出来都是具有层次的,因此也被称为层次结构。

3. 相关术语

通常,在逻辑上表达一棵抽象的树状结构的时候,习惯于将树根放在顶部,树枝树杈向下生长,如下 图所示。

对于一棵树来说,有如下基本术语:

1. 结点:

树中的元素 及其子树

2. 根(root):

树的第一个节点,没有直接前驱。如上图中的A。

3. 双亲节点/父节点(parent):

某节点的直接前驱称为该节点的双亲节点,或成为父节点。例如上图中A是B的父节点。

4. 孩子节点/子节点(child):

某节点的直接后继称为该节点的孩子节点。例如上图中B、C、D均为A的孩子节点。

5. 节点的层次(level):

根节点所在的层次规定为第1层,其孩子所在的层次为第2层,后代节点以此类推。比如上图中节 点E的层次是3。

6. 节点的度(degree):

一个节点拥有的孩子节点的总数,称为该节点的度。比如上图中节点B的度为2。

7. 叶子(leaf):

一棵树中度等于0的节点,被称为叶子,又称为终端节点。比如上图中K、L、F、G、M、I、J均为 叶子。

8. 树的高度(height):

一棵树中所有节点的层次的最大值,称为这棵树的高度,又称为树的深度。比如上图的树的高度为 4。

9. 有序树与无序树:

一棵树中,如果某个节点的孩子节点之间是有次序的,则称这棵树为有序树,反之称为无序树。


4. 二叉树

在各种不同的树状结构中,最常见也最重要的是二叉树(Binary Tree),下面是二叉树的定义:

有序树

任意节点的度小于等于2

比如如下这棵树就是一棵二叉树。其中8是根节点,14是10的右孩子(因为二叉树是有序树,因此严 格区分左右),而13则是14的左孩子。

为了方便对二叉树进行操作,通常会对一棵它进行标号:从上到下,从左到右进行标号:

注意:

没有孩子节点的地方也要标出来

对于二叉树而言,有如下特性:

1. 第𝑖层上,最多有2𝑖−1个节点。

2. 高度为𝑘的二叉树,最多有2𝑘−1个节点。

3. 假设叶子数目为𝑛0,度为2的节点数目为𝑛2,则有:

二叉树的一般结构:

满二叉树

一棵深度为k,且有2^k-1个结点的二叉树,称为满二叉树。这种树的特点是每一层上的结点数都是 最大结点数。

简单理解: 除了叶子节点之外,其余节点的度都为2;其特点是: 如果深度为 K,则节点数为 2^K - 1。

完全二叉树

在一棵二叉树中,除最后一层外,若其余层都是满的,或者最后一层是满的,或者是最后一层在右 边缺少连续若干结点,则此二叉树为完全二叉树。

简单理解:除最后一层叶子节点外。是一颗满二叉树,最后一层由右向左有连续缺省的0个,1个或 多个节点。

二叉搜索树(BST) 特点:

1. 如果节点具有左子树,则左子树上所有节点都不大于该节点的值;

2. 节点具有右子树,则右子树上所有节点都不小于该节点的值;

3. 子树又是二叉搜索数

二叉搜索树(BST)

二叉树(BST)的组成

根指针:指向根节点的指针变量

节点:

数据域 (存储的实际数据)

指针域 (左,右指针)

结构设计:

typedef int data_t;
typedef struct _node
{
data_t data; // 数据域
struct _node *left; // 左子树指针
struct _node *right;// 右子树指针
}NODE;

二叉树(BST)的算法

创建二叉树

int btree_create(NODE** root,data_t data);

二叉树数据添加

int btree_add(NODE** root,data_t data);

示例图:

二叉树数据遍历

前序遍历(先序遍历,即 根左右 ))

void Preorder(const NODE* root);

前序遍历通俗的说就是从二叉树的根结点出发,先输出根结点数据,然后输出左结点,最后输 出右结点的数据。

从根结点出发,则第一次到达结点A,故输出A;继续向左访问,第一次访问结点B,故输出B; 按照同样规则,输出D,输出H;当到达叶子结点H,返回到D,此时已经是第二次到达D,故不 在输出D,进而向D右子树访问,D右子树不为空,则访问至I,第一次到达I,则输出I;I为叶子 结点,则返回到D,D左右子树已经访问完毕,则返回到B,进而到B右子树,第一次到达E,故 输出E;向E左子树,故输出J;按照同样的访问规则,继续输出C、F、G。

前序遍历输出结果:ABDHIEJCFG 中序

遍历(即 左根右 )

void Midorder(const NODE* root);

中序遍历通俗的说就是从二叉树的根结点出发,先输出左结点数据,然后输出根结点,最后输 出右结点的数据。

从根结点出发,则第一次到达结点A,不输出A,继续向左访问,第一次访问结点B,不输出 B;继续到达D,H;

到达H,H左子树为空,则返回到H,此时第二次访问H,故输出H;H右子树为空,则返回至 D,此时第二次到达

D,故输出D;由D返回至B,第二次到达B,故输出B;按照同样规则继续访问,输出J、E、A、 F、C、G;

中序遍历输出结果:HDIBJEAFCG

后序遍历(即 左右根 )

void Postorder(const NODE* root);

后序遍历通俗的说就是从二叉树的根结点出发,先输出左结点数据,然后输出右结点,最后输 出根结点的数据。

从根结点出发,则第一次到达结点A,不输出A,继续向左访问,第一次访问结点B,不输出 B;继续到达D,H;

到达H,H左子树为空,则返回到H,此时第二次访问H,不输出H;H右子树为空,则返回至 H,此时第三次到达

H,故输出H;由H返回至D,第二次到达D,不输出D;继续访问至I,I左右子树均为空,故第 三次访问I时,输出

I;返回至D,此时第三次到达D,故输出D;按照同样规则继续访问,输出J、E、B、F、G、 C,A;

后序遍历输出为: HIDJEBFGCA

层序遍历

 void Levelorder(const NODE* root);

二叉树数据查询

NODE* btree_find(const NODE* root,data_t data);

① 从根结点出发

② 如果比根节点小,那么就去其左子树找

③ 如果比根节点大就去其右子树找

④ 找到叶子都没找到, 就代表查找失败

二叉树数据更新

int btree_update(const NODE* root,data_t old,data_t newdata);

二叉树回收

void btree_destroy(NODE** root);

二叉树数据删除

int btree_delete(NODE** root,data_t data);

原则:将待删除的节点尽量转换为删除叶子节点,因为删除叶子节点对BST树影响是最小的;

思路:

① 从根节点开始遍历BST找到待删除的节点;

② 对待删除的节点状态进行判断,如果节点有左子树,找到左子树中最大的节点,然后利用 左子树中最大的节 点数据替换待删除的节点数据,删除左子树中最大的节点;左子树中最大的节点大概率 是叶子节点。

③ 如果节点只有右子树,找到右子树中最小的节点,然后 利用右子树中最小的节点数据替 换待删除的节点数 据,删除右子树中最小的节点;右子树中最小的节点大概率是叶子节点。

④ 如果待删除节点是叶子节点,直接删除。

二叉树(BST)完整实现

队列实现

SQueue.h

#ifndef __SQUEUE_H
#define __SQUEUE_H
#include "btree.h"
typedef NODE* type_t;
typedef struct
{
type_t *pData;
int size;
int head;
int tail;
}SQueue;
int SQ_init(SQueue *q, int num);
int SQ_isfull(SQueue *q);
int SQ_isempty(SQueue *q);
int SQ_push(SQueue *q,type_t data);
int SQ_pop(SQueue *q,type_t *data);
int SQ_free(SQueue *q);
#endif



SQueue.c

#include <stdlib.h>
#include "SQueue.h"
int SQ_init(SQueue* q,int num)
{
q -> pData = (type_t*)calloc(sizeof(type_t),num);
if(q -> pData == NULL)
return -1;
q -> size = num ;
q -> head = q -> tail = 0;
return 0;
}
int SQ_isfull(SQueue *q)
{
return (q -> tail + 1) % q -> size == q -> head;
}
int SQ_isempty(SQueue *q)
{
return q -> tail == q -> head;
}
int SQ_push(SQueue *st,type_t data)
{
if(SQ_isfull(st))
return -1;
st -> pData[st -> tail] = data;
st -> tail = (st -> tail+1) % st -> size;
return 0;
}
int SQ_pop(SQueue *st,type_t *data)
{
if(SQ_isempty(st))
return -1;
*data = st -> pData[st -> head];
st -> head = (st -> head+1) % st -> size;
return 0;
}
int SQ_free(SQueue *st)
{
if(st -> pData)
{
free(st->pData);
st -> pData = NULL;
}
st -> head = st -> tail = 0;
}

树的实现

btree.h

#ifndef __BTREE_H
#define __BTREE_H
typedef int data_t;
typedef struct _node
{
data_t data; // 节点上的数据
struct _node *left; // 该节点左侧子节点的地址
struct _node *right;// 该节点右侧子节点的地址
}NODE;
// 创建搜索二叉树
int btree_create(NODE** root,data_t data);
// 二叉树数据添加
int btree_add(NODE** root,data_t data);
// 二叉树数据删除
int btree_delete(NODE** root,data_t data);
// 二叉树前序遍历
void Preorder(const NODE* root);
// 二叉树中序遍历
void Midorder(const NODE* root);
// 二叉树后序遍历
void Postorder(const NODE* root);
// 二叉树层序遍历
void Levelorder(const NODE* root);
// 二叉树数据查询
NODE* btree_find(const NODE* root,data_t data);
// 更新二叉树数据old 为 newdata
int btree_update(const NODE* root,data_t old,data_t newdata);
// 二叉树回收
void btree_destroy(NODE** root);
#endif

btree.c

#include "btree.h"
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include "SQueue.h"
/*
@function: int btree_create(NODE** root,data_t data)
@brief: 创建搜索二叉树
@argument: root: 根指针地址
@argument: data: 存储的数据
@ret : 0 成功
-1 失败
*/
int btree_create(NODE** root,data_t data)
{
if(*root)
return -1;
NODE* p = (NODE*)malloc(sizeof(NODE));
if(!p)
return -1;
p -> data = data;
p -> left = NULL;
p -> right = NULL;
*root = p;
return 0;
}
/*
@function: int btree_add(NODE** root,data_t data)
@brief: 二叉树数据添加
@argument: root: 根指针地址
@argument: data: 添加的数据
@ret : 0 成功
-1 失败
*/
int btree_add(NODE** root,data_t data)
{
NODE* pNew = (NODE*)malloc(sizeof(NODE));
if(!pNew)
return -1;
pNew -> data = data;
pNew -> left = NULL;
pNew -> right = NULL;
NODE* p = *root, *q = NULL;
if(!p)
{
*root = pNew;
return 0;
}
while(p)
{
q = p;
if(memcmp(&data,&(p -> data),sizeof(data_t)) < 0)
p = p -> left;
else
p = p -> right;
}
if(memcmp(&data,&(q -> data),sizeof(data_t)) < 0)
q -> left = pNew;
else
q -> right = pNew;
return 0;
}
/*
@function: int btree_delete(NODE** root,data_t data)
@brief: 二叉树数据删除
@argument: root: 根指针地址
@argument: data: 待删除的节点数据
@ret : 0 成功
-1 失败
*/
int btree_delete(NODE** root,data_t data)
{
/*
原则:将待删除的节点尽量转换为删除叶子节点,因为删除叶子节点对BST树影响是最小的;
思路:
1. 从根节点开始遍历BST找到待删除的节点;
2. 对待删除的节点状态进行判断,如果节点有左子树,找到左子树中最大的节点,然后利用左子树中最大的
节点数据替换待删除 的节点数据,删除左子树中最大的节点;左子树中最大的节点大概率是叶子节点
3. 如果节点只有右子树,找到右子树中最小的节点,然后 利用右子树中最小的节点数据替换待删除的节点
数据,删除右子树中 最小的节点;右子树中最小的节点大概率是叶子节点
4. 如果待删除节点是叶子节点,直接删除。
*/
NODE* del = *root; //指向待删除的节点
NODE* parent = NULL; //指向实际删除节点的双亲节点
NODE* replace = NULL; //指向实际删除的节点
while(del)
{
if(memcmp(&(del -> data),&data,sizeof(data_t)) < 0) //待删除的数据大于当前节点数据,向右子
树继续查找
{
parent = del;
del = del -> right;
}
else if(memcmp(&(del->data),&data,sizeof(data_t)) > 0) //待删除的数据小于当前节点数据,向
左子树继续查找
{
parent = del;
del = del -> left;
}
else // 找到了待删除的节点
{
if(del -> left) //待删除的节点有左子树
{
parent = del ;
replace = del -> left;
while(replace -> right) //找左子树中最大的节点
{
parent = replace;
replace = replace -> right;
}
del -> data = replace -> data;
if(parent -> right == replace) //实际删除的节点在双亲节点的右子树上
parent -> right = replace -> left;
else
parent -> left = replace -> left;
free(replace);
}
else if(del -> right) //待删除的节点只有右子树
{
parent = del;
replace = del -> right;
while(replace -> left)
{
parent = replace;
replace = replace -> left;
}
del -> data = replace -> data;
if(parent -> right == replace)
parent -> right = replace -> right;
else
parent -> left = replace -> right;
free(replace);
}
else //待删除的节点是叶子
{
if(!parent)
{
free(del);
*root = NULL;
return 0;
}
if(parent -> left == del)
parent -> left = NULL;
else
parent -> right = NULL;
free(del);
}
return 0;
}
}
return -1;
}
/*
@function: void Preorder(const NODE* root)
@brief: 二叉树前序遍历
@argument: root: 根指针
@ret : 无
*/
void Preorder(const NODE* root)
{
if(!root)
return ;
printf("%4d",root -> data);
Preorder(root->left);
Preorder(root->right);
}
/*
@function: void Midorder(const NODE* root)
@brief: 二叉树中序遍历
@argument: root: 根指针
@ret : 无
*/
void Midorder(const NODE* root)
{
if(!root)
return ;
Midorder(root->left);
printf("%4d",root -> data);
Midorder(root->right);
}
/*
@function: void Postorder(const NODE* root)
@brief: 二叉树后序遍历
@argument: root: 根指针
@ret : 无
*/
void Postorder(const NODE* root)
{
if(!root)
return ;
Postorder(root->left);
Postorder(root->right);
printf("%4d",root -> data);
}
/*
@function: void Postorder(const NODE* root)
@brief: 二叉树层序遍历
@argument: root: 根指针
@ret : 无
*/
void Levelorder(const NODE* root)
{
SQueue q;
SQ_init(&q,100);
SQ_push(&q,(type_t)root);
while(!SQ_isempty(&q))
{
type_t dt;
if(0 == SQ_pop(&q,&dt))
{
printf("%4d",dt -> data);
if(dt -> left)
SQ_push(&q,dt->left);
if(dt -> right)
SQ_push(&q,dt->right);
}
}
printf("\n");
SQ_free(&q);
}
/*
@function: NODE* btree_find(const NODE* root,data_t data)
@brief: 二叉树数据查询
@argument: root: 根指针
@argument: data: 待查询的数据
@ret : 成功:返回查询到的节点地址
失败: NULL
*/
NODE* btree_find(const NODE* root,data_t data)
{
const NODE *p = root;
while(p)
{
if(memcmp(&data,&(p -> data),sizeof(data_t)) < 0)
p = p -> left;
else if(memcmp(&data,&(p -> data),sizeof(data_t)) > 0)
p = p -> right;
else
return (NODE*)p;
}
return NULL;
}
/*
@function: int btree_update(const NODE* root,data_t old,data_t newdata)
@brief: 更新二叉树数据old 为 newdata
@argument: root: 根指针
@argument: old: 待更新的数据
@argument: newdata: 更新后的数据
@ret : 成功:0
失败: -1
*/
int btree_update(const NODE* root,data_t old,data_t newdata)
{
NODE* pfind = btree_find(root,old);
if(!pfind)
return -1;
pfind -> data = newdata;
return 0;
}
/*
@function: void btree_destroy(NODE** root)
@brief: 二叉树回收
@argument: root: 根指针地址
@ret : 无
*/
static void btree_free(NODE* root)
{
if(!root)
return ;
btree_free(root->left);
btree_free(root->right);
free(root);
}
void btree_destroy(NODE** root)
{
btree_free(*root);
*root = NULL;
}

btree_main.c

#include "btree.h"
#include <stdio.h>
#include <time.h>
#include <stdlib.h>
#define N 10
int main(void)
{
NODE *root = NULL;
register int i = 0;
data_t a[] = {20,15,25,10,19,28,6,13,26,30,11,27};
int n = sizeof a / sizeof a[0];
srand(time(NULL));
for(; i < n ; i++)
{
// int data = rand() % 99 + 1;
// printf("%4d",data);
// btree_add(&root,data);
btree_add(&root,a[i]);
}
printf("\n");
puts("====先序遍历====");
Preorder(root);
printf("\n");
puts("====中序遍历====");
Midorder(root);
printf("\n");
puts("====后序遍历====");
Postorder(root);
printf("\n");
puts("====层序遍历====");
Levelorder(root);
printf("\n");
data_t deldata = 0;
while(1)
{
printf("请输入要删除的数据(-1退出):");
scanf("%d",&deldata);
if(deldata == -1)
break;
if(btree_delete(&root,deldata) == -1)
{
puts("删除失败请重试...");
continue;
}
puts("====先序遍历====");
Preorder(root);
printf("\n");
}
btree_destroy(&root);
return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2085785.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[报错] nvcc -V 找不到

报错&#xff1a; nvcc : 无法将“nvcc”项识别为 cmdlet、函数、脚本文件或可运行程序的名称。请检查名称的拼写&#xff0c;ObjectNotFound: (nvcc:String) [], CommandNotFoundExceptionFullyQualifiedErrorId : CommandNotFoundException 找不到 nvcc -V&#xff0c;试过…

C语言基础(二十四)

堆栈&#xff08;Stack&#xff09;是一种遵循后进先出&#xff08;LIFO, Last In First Out&#xff09;原则的数据结构。堆栈的主要操作包括压栈&#xff08;Push&#xff09;、弹栈&#xff08;Pop&#xff09;、查看栈顶元素&#xff08;Peek/Top&#xff09;等。C语言标准…

Qt折线图,x轴是时间,y轴是int

头文件 #include <QWidget> #include <QTimer> #include <QtCharts/QChartView>//显示图表 #include <QtCharts/QLineSeries>//线系列 #include <QtCharts/QSplineSeries> #include <QtCharts/QValueAxis> #include <QDateTime> #i…

[RIS]GRES: Generalized Referring Expression Segmentation

1. BaseInfo TitleGRES: Generalized Referring Expression SegmentationAdresshttps://arxiv.org/pdf/2306.00968Journal/TimeCVPR2023Author南洋理工Codehttps://github.com/henghuiding/ReLARead20240829TableVisonLanguage 2. Creative Q&A 考虑结果多目标和无目标&…

skywalking接入nginx

下载Nginx的插件 GitHub - apache/skywalking-nginx-lua: The Nginx Lua agent for Apache SkyWalking ~]# cd skywalking-nginx-lua-master/ skywalking-nginx-lua-master]# ls CHANGES.md examples lib LICENSE licenses Makefile mvnw NOTICE README.md RELEASING…

苹果手机系统修复如何操作,几种iOS系统修复办法分享

苹果手机作为市场上的热门产品&#xff0c;其系统稳定性和用户体验一直备受赞誉。然而&#xff0c;即使是如此优秀的系统&#xff0c;也难免会遇到一些问题&#xff0c;如系统崩溃、应用闪退、无限重启等。当这些问题出现时&#xff0c;如何进行系统修复成为了许多用户关心的问…

@antv/x6 自定义节点Safari兼容问题处理

背景 为什么选择antv/x6&#xff1f; 由于x6提供了一套强大友好的流程图API&#xff0c;并且支持使用 React、Vue 组件来渲染节点。这样可以使用组件开发的方式去快速完成卡片开发&#xff0c;并实现更加复杂的业务逻辑。遇到了Safari的兼容问题 由于x6自定义节点基于SVG fore…

Milvus向量数据库-数据备份与恢复

前言 随着Milvus版本的持续迭代&#xff0c;越来越多的用户将其作为构建生产环境的向量数据服务使用。作为数据服务使用&#xff0c;其中的运维、数据安全、容灾备份自然是用户最关心且不容有失的需求。为解决这一需求&#xff0c;Milvus-backup项目工具应运而生。 Milvus-ba…

docker实战演练

一.docker简介 二.docker部署 1.解压并安装所需要的软件 [rootdocker-node1 ~]# ls anaconda-ks.cfg docker.tar.gz mario.tar.gz busybox-latest.tar.gz game2048.tar.gz nginx-latest.tar.gz [rootdocker-node1 ~]# tar zxf docker.tar.gz [rootdocker-node1 …

cad转换dxf怎么转换?学会这5种方法轻松解决cad难题

在CAD领域中&#xff0c;DXF格式因其良好的兼容性和广泛的应用&#xff0c;成为了许多设计师和工程师的首选交换格式。然而&#xff0c;CAD文件很多默认的都是dwg格式&#xff0c;那么如何将CAD文件转换为DXF格式&#xff0c;很多小伙伴不是很清楚&#xff0c;下面给大家分享5种…

想学gis开发,java和c++哪个比较好?

java与C的应用场景不同&#xff0c;究竟选择谁&#xff0c;应该由开发者的兴趣方向来决定。 你选择Java&#xff0c;意味着以后的业务方向就是偏后台服务开发&#xff0c;如果你非得说我用java也可以写界面&#xff0c;对不起&#xff0c;别人不会。不管是从项目还是产品的角度…

Element-plus组件库基础组件使用

文章目录 按钮图标输入框表格表单对话框文件上传布局容器ElMessage消息提示MessageBox 消息提示框Pagination分页样式表单校验 记录vue3项目使用element-Plus&#xff0c;开发中常用的一些样式 下面这些组件是写增删改查是经常用到的&#xff0c;学习了这个写增删改查会好很多。…

DBpedia——利用维基百科信源构造语义知识库

1. 简介 DBpedia 从维基百科(Wikipedia)的词条里撷取出结构化的信息&#xff0c;以强化维基百科的搜寻功能&#xff0c;同时还提供复杂信息的检索功能&#xff0c;并将其他资料集连结至维基百科。通过这样的语义化技术的介入&#xff0c;让维基百科的庞杂资讯有了许多创新而有趣…

iview checkbox单独使用时 如何去掉显示的true和false以及不显示文字

如图所有 当在表格显示中想单独使用checkbox的时候&#xff0c;只想显示勾选 不显示文字这样写&#xff0c;分别用2个参数true/false来显示勾选和不勾选&#xff0c;再在外层用参数控制就好啦&#xff0c;是不是很简单呢&#xff01;

各行各业的都有的文档推荐

如果你在寻找一个丰富多样、几乎能满足任何阅读需求的电子书下载的地方&#xff0c;不妨考虑“literalink.top”网站。它不仅涵盖了广泛的书籍种类&#xff0c;从经典文学到最新畅销书应有尽有&#xff0c;还提供了多本不同版本和译本的选择&#xff0c;超越了许多其他网站的资…

GaussDB 24.1.30 分布式3节点命令行方式部署

目录 GaussDB介绍 服务器环境 安装前准备 配置会话不中断 操作系统配置 关闭防火墙并禁止开机启动 设置时区和时间 检查时区和时间 java版本 expect root密码一致 root用户ssh连通性 上传软件包和安装脚本 安装脚本配置 修改 install_cluster.json 配置文件 安装…

2024最新最全面的JMeter 做接口加密测试

JMeter 怎么做接口的加密&#xff1f; JMeter如果需要做加密测试&#xff0c;是需要加密类型对应的jar包的。本文以MD5,加密作为教程。 1、在Test Plan 引用jar包&#xff1b; 2、添加BeanShell Sampler取样器&#xff0c;并输入调用代码 import md5.mymd5;//调用jar包 String…

使用kafka完成数据的实时同步,同步到es中。(使用kafka实现自动上下架 upper、lower)

文章目录 1、发送消息 KafkaService2、生产者 service-album -> AlbumInfoServiceImpl2.1、新增 saveAlbumInfo()2.2、更新 updateAlbumInfo()2.3、删除 removeAlbumInfo() 3、消费者 service-search - > AlbumListener.java 上架&#xff1a;新增专辑到 es下架&#xff…

信息学奥赛初赛天天练-78-NOIP2015普及组-基础题3-中断、计算机病毒、文件传输协议FTP、线性表、链式存储、栈

NOIP 2015 普及组 基础题3 8 所谓的“中断”是指( ) A 操作系统随意停止一个程序的运行 B 当出现需要时&#xff0c;CPU 暂时停止当前程序的执行转而执行处理新情况的过程 C 因停机而停止一个程序的运行 D 电脑死机 9 计算机病毒是( ) A 通过计算机传播的危害人体健康的一种病…

有什么还原空白试卷的免费软件?2024快速进行空白还原的软件

有什么还原空白试卷的免费软件&#xff1f;2024快速进行空白还原的软件 还原空白试卷通常是指将已经填写的试卷还原为未填写的空白版本&#xff0c;以便重复使用或进行其他用途。以下是五款可以帮助你快速还原空白试卷的免费软件&#xff0c;这些工具提供了不同的功能&#xf…