【图解数据结构】顺序表实战指南:手把手教你详细实现(超详细解析)
🌈个人主页:聆风吟
🔥系列专栏:图解数据结构、算法模板
🔖少年有梦不应止于心动,更要付诸行动。
文章目录
- 一. ⛳️线性表
- 1.1 🔔线性表的定义
- 1.2 🔔线性表的存储结构
- 二. ⛳️顺序表
- 2.1 🔔顺序表定义
- 2.2 🔔顺序表的分类
- 2.2.1 👻静态顺序表
- 2.2.2 👻动态顺序表
- 三. ⛳️顺序表的基本操作实现
- 3.1 🔔动态顺序表结构体构建
- 3.2 🔔初始化顺序表
- 3.3 🔔销毁顺序表
- 3.4 🔔打印顺序表
- 3.4 🔔扩容
- 3.5 🔔尾插
- 3.6 🔔尾删
- 3.7 🔔头插
- 3.8 🔔头删
- 3.9 🔔在下标为pos位置插入x
- 3.10 🔔删除下标为pos位置的数据
- 3.11 🔔查找某个值的下标
- 四. ⛳️顺序表的完整源代码
- 4.1 🔔SeqList.h 顺序表的函数声明
- 4.2 🔔SeqList.c 顺序表的函数定义
- 4.3 🔔test.c 顺序表功能测试
- 📝总结
一. ⛳️线性表
1.1 🔔线性表的定义
线性表(linear list):线性表是一种数据结构,由n个具有相同数据类型的元素构成一个有限序列。线性表可以用数组、链表、栈等方式实现,常见的线性表有数组、链表、栈、队列等,也可以自定义实现。
这里需要强调一下几点:
首先它是一个序列。也就是说,元素之间是有顺序的。线性表中的元素称为结点,相邻结点之间的关系称为邻接关系。除第一个结点无前驱和最后一个结点无后继外,其他每个结点有且仅有一个前驱和一个后继。图解如下:
注意:
线性表元素个数n (n >= 0)
定义为线性表的长度,当n=0
时,称为空表。
1.2 🔔线性表的存储结构
线性表的存储结构有顺序存储结构和链式存储结构两种。前者称为顺序表,后者称为链表:
其中,线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构上并不一定是连续的,线性表在物理上存储时,通常以数组和链式结构的形式存储。
本文主要详细讲解线性表的顺序存储结构 —— 顺序表。线性表的链式存储将在下期讲解,言归正传接下来让我们开始今天的 “主菜" 学习。
二. ⛳️顺序表
2.1 🔔顺序表定义
顺序表(Sequential List):用一段物理地址连续的存储单元依次存储数据元素的线性结构。一般情况下采用数组存储。在数组上完成数据的增删查改。
2.2 🔔顺序表的分类
顺序表一般可以分为:静态顺序表和动态顺序表。
2.2.1 👻静态顺序表
静态顺序表:指存储空间是固定的并且在程序运行前就已经确定大小的顺序表。它通常使用数组来实现,即通过定义一个固定长度的数组来存储数据元素。
静态顺序表的结构代码:
//静态顺序表 —— 使用定长数组存储元素(不实用)
#define MAXSIZE 7//存储单元初始分配量
typedef int SLDataType;//SLDataType类型根据实际情况而定,这里是inttypedef struct SeqList
{SLDataType data[MAXSIZE];//定长数组int size;//有效数据的个数
}SeqList;
我们可以发现描述静态顺序表需要三个属性:
- 存储空间的起始位置:数组
data
,他的存储位置就是存储空间的存储位置; - 线性表的最大存储容量:数组长的
MAXSIZE
; - 线性表的当前位置:
size
。
静态顺序表的优缺点:
- 由于静态顺序表大小是固定的,因此不支持动态插入和删除,但可以通过重新分配空间的方式来增加或减少容量;
- 静态顺序表的优点:访问数据快速,由于是连续存储,所以可以直接通过下标访问元素,效率高;
- 静态顺序表的缺点:空间利用率低,因为必须预留足够的空间,以防止溢出。
2.2.2 👻动态顺序表
动态顺序表:通过动态分配内存空间,实现随着数据量的增加而不断扩容的效果。它的结构类似于一个数组,数据元素的存储是连续的,支持随机访问和顺序访问。
动态顺序表的结构代码:
//动态顺序表 —— 使用动态开辟的数组存储
typedef int SLDataType;//SLDataType类型根据实际情况而定,这里是inttypedef struct SeqList
{SLDataType* a;//指向动态开辟的数组int size;//有效数据的个数int capacity;//记录容量的空间大小
}SL;
我们可以发现描述动态顺序表也需要三个属性:
- 存储空间的起始位置:指针
a
,他里面存储的地址就是存储空间的地址; - 线性表当前最大存储容量:
capacity
,可以通过动态分配的方式进行扩容; - 线性表的当前位置:
size
。
动态顺序表的优缺点:
- 动态顺序表的优点:可以使用指针动态地分配内存,具有高效的存储和访问效率;
- 动态顺序表的缺点:在插入和删除元素时需要移动大量的数据,效率较低。
三. ⛳️顺序表的基本操作实现
通过上面的学习我们已经初步了解静态顺序表和动态顺序表,有同学估计要问了在日常生活中我们应该使用哪种呢?在这里作者推荐大家使用动态顺序表。因为动态顺序表可以使程序更加高效和灵活,可以根据实际数据量动态地调整表的大小,避免在创建静态顺序表时浪费内存空间或者当数据量超出静态顺序表容量时造成数据丢失或程序崩溃等问题。本文也将采用动态顺序表结合图文去实现顺序表的基本操作。
3.1 🔔动态顺序表结构体构建
//动态顺序表
#define SLCAPACITY 4
typedef int SLDataType;typedef struct SeqList
{SLDataType* a;//指向动态开辟的数组int size;//有效数据的个数int capacity;//记录容量的空间大小
}SL;
代码深剖:
- 结构体中
a
指向的数组类型是我们重定义的SLDataType,这样当我们想创建其它类型的顺序表时只需要对typedef
后面的类型进行需改即可; size
是用来计数的,统计当前顺序表一共有多少个有效元素;capacity
是用来表示当前顺序表的容量,当size==capacity
时说明当前顺序表已经“装满了”,需要扩容;- 定义标识符
SLCAPACITY
,方便后文对顺寻表进行初始化可以方便改变capacity
的初始值。
3.2 🔔初始化顺序表
//初始化顺序表
void SLInit(SL* ps)
{assert(ps);//使用malloc开辟空间ps->a = (SLDataType*)malloc(sizeof(SLDataType)*4);//判断空间是否开辟成功if (NULL == ps->a){perror("malloc failed");exit(-1);}ps->size = 0;ps->capacity = SLCAPACITY;
}
代码深剖:
- 在这里我们需要使用
assert
对ps
进行一下断言,以防传入空指针(后文在出现就不多做叙述了)。 - 使用
malloc
开辟空间,一定要进行判断是否开辟成功,如果不进行判断直接使用可能会导致程序崩溃。
时间复杂度:
该程序没有循环,根据大O阶的推导方法很容易得出:初始化顺序表的时间复杂度为
O(1)
3.3 🔔销毁顺序表
//销毁顺序表
void SLDestroy(SL* ps)
{assert(ps);free(ps->a);ps->a = NULL;ps->size = ps->capacity = 0;
}
代码深剖:
为什么在这里要销毁顺序表呢?因为我们在这里使用的动态顺序表,a
是通过malloc
进行动态申请的空间,如果使用了malloc分配的内存空间后忘记释放,会导致内存泄漏,浪费系统资源甚至导致程序崩溃。
时间复杂度:
该程序没有循环,根据大O阶的推导方法很容易得出:销毁顺序表的时间复杂度为
O(1)
3.4 🔔打印顺序表
//打印顺序表
void SLPrint(SL* ps)
{assert(ps);for (int i = 0; i < ps->size; i++){printf("%d ", ps->a[i]);}printf("\n");
}
代码深剖:
打印顺序表就是进行简单的遍历循环,此处不多做叙述。
时间复杂度:
该程序有单层循环,根据大O阶的推导方法很容易得出:打印顺序表的时间复杂度为
O(n)
3.4 🔔扩容
因为扩容在尾插、头插以及在pos位置插入都需要使用,因此我们可以把扩容单独封装成一个函数,可以降低代码的的冗余。整体思路图解:
//检查容量是否够,不够进行扩容
void SLCheckCapacity(SL* ps)
{assert(ps);//满了要扩容if (ps->size == ps->capacity){//使用realloc进行扩容SLDataType* temp = (SLDataType*)realloc(ps->a, sizeof(SLDataType) * 2 * (ps->capacity));//检查是否扩容成功if (temp == NULL){perror("realloc failed");exit(-1);}ps->a = temp;ps->capacity *= 2;}
}
代码深剖:
realloc
是C语言中的一个函数,用于重新分配已经分配的内存空间的大小。它的原型是:
//头文件
#include<stdlib.h>
//原型
extern void *realloc(void *mem_address, unsigned int newsize)
其中,mem_address
是指向已分配内存的指针,newsize
是新的内存大小。如果内存分配失败,将会会返回NULL
。
时间复杂度:
该程序没有循环,根据大O阶的推导方法很容易得出:扩容的时间复杂度为
O(1)
3.5 🔔尾插
尾插时需要先判断顺序表是否满了,满了要先进行扩容才能继续进行扩容。size
表示有效元素个数,同时也是顺序表中最后一个元素后一个位置的下标。成功插入后要对有效数据个数size
进行加1操作。整体思路图解:
//尾插
void SLPushBack(SL* ps, SLDataType x)
{assert(ps);//检查是否需要扩容SLCheckCapacity(ps);ps->a[ps->size] = x;ps->size++;
}
时间复杂度:
该程序没有循环,根据大O阶的推导方法很容易得出:尾插的时间复杂度为
O(1)
3.6 🔔尾删
整体思路图解:
//尾删
void SLPopBack(SL* ps)
{assert(ps);//温柔检查/*if (ps->size == 0)return;*///暴力检查assert(ps->size > 0);ps->size--;
}
代码深剖:
在代码中我们提供两种检查顺序表是否为空的办法。第一种是比较温柔的检查,如果顺序表为空直接返回,返回之后仍然可以进行其他操作。第二种是比较暴力的检查方法,直接提示错误并打印出错误位置的行号。
时间复杂度:
该程序没有循环,根据大O阶的推导方法很容易得出:尾删的时间复杂度为
O(1)
3.7 🔔头插
整体思路图解:
//头插
void SLPushFront(SL* ps, SLDataType x)
{assert(ps);//检查是否需要扩容SLCheckCapacity(ps);//挪动数据int end = ps->size - 1;while (end >= 0){ps->a[end + 1] = ps->a[end];--end;}ps->a[0] = x;ps->size++;
}
时间复杂度:
该程序需要执行单层循环,根据大O阶的推导方法很容易得出:头插的时间复杂度为
O(n)
3.8 🔔头删
整体思路图解:
//头删
void SLPopFront(SL* ps)
{assert(ps);//判断顺序表是否为空assert(ps->size > 0);//挪动元素向前覆盖int begin = 1;while (begin < ps->size){ps->a[begin - 1] = ps->a[begin];++begin;}ps->size--;
}
时间复杂度:
该程序需要执行单层循环,根据大O阶的推导方法很容易得出:头插的时间复杂度为
O(n)
3.9 🔔在下标为pos位置插入x
整体思路图解:
//在下标为pos的位置插入x
void SLInsert(SL* ps, int pos, SLDataType x)
{assert(ps);//检查pos是否在有效范围内assert(pos >= 0 && pos <= ps->size);//检查是否需要扩容SLCheckCapacity(ps);//挪动数据int end = ps->size - 1;while (end >= pos) {ps->a[end + 1] = ps->a[end];--end;}//插入元素ps->a[pos] = x;ps->size++;
}
时间复杂度:
该程序需要执行单层循环,根据大O阶的推导方法很容易得出:pos位置插入的时间复杂度为
O(n)
3.10 🔔删除下标为pos位置的数据
整体思路图解:
//删除下标为pos位置的数据
void SLErase(SL* ps, int pos)
{assert(ps);//检查pos是否在有效范围内assert(pos >= 0 && pos < ps->size);//挪动元素向前覆盖int begin = pos + 1;while (begin < ps->size){ps->a[begin - 1] = ps->a[begin];++begin;}ps->size--;
}
时间复杂度:
该程序需要执行单层循环,根据大O阶的推导方法很容易得出:pos位置删除的时间复杂度为
O(n)
3.11 🔔查找某个值的下标
//查找某个值的下标,没找到返回-1
int SLFind(SL* ps, SLDataType x)
{assert(ps);for (int i = 0; i < ps->size; i++){if (ps->a[i] == x)return i;}return -1;
}
时间复杂度:
该程序需要执行单层循环,根据大O阶的推导方法很容易得出:查找的时间复杂度为
O(n)
四. ⛳️顺序表的完整源代码
4.1 🔔SeqList.h 顺序表的函数声明
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>//动态顺序表
#define SLCAPACITY 4
typedef int SLDataType;typedef struct SeqList
{SLDataType* a;//指向动态开辟的数组int size;//有效数据的个数int capacity;//记录容量的空间大小
}SL;//管理数据 —— 增删查改//初始化
void SLInit(SL* ps);
//销毁顺序表
void SLDestroy(SL* ps);
//打印顺序表
void SLPrint(SL* ps);
//检查容量是否够,不够进行扩容
void SLCheckCapacity(SL* ps);//尾插尾删
void SLPushBack(SL* ps, SLDataType x);
void SLPopBack(SL* ps);//头插头删
void SLPushFront(SL* ps, SLDataType x);
void SLPopFront(SL* ps);//查找某个值的下标,没找到返回-1
int SLFind(SL* ps, SLDataType x);//在pos位置插入x
void SLInsert(SL* ps, int pos, SLDataType x);
//删除pos位置的数据
void SLErase(SL* ps, int pos);
4.2 🔔SeqList.c 顺序表的函数定义
#include "SeqList.h"//初始化顺序表
void SLInit(SL* ps)
{assert(ps);//使用malloc开辟空间ps->a = (SLDataType*)malloc(sizeof(SLDataType)*4);//判断空间是否开辟成功if (NULL == ps->a){perror("malloc failed");exit(-1);}ps->size = 0;ps->capacity = SLCAPACITY;
}//销毁顺序表
void SLDestroy(SL* ps)
{assert(ps);//释放动态开辟的空间free(ps->a);ps->a = NULL;ps->size = ps->capacity = 0;
}//打印顺序表
void SLPrint(SL* ps)
{assert(ps);for (int i = 0; i < ps->size; i++){printf("%d ", ps->a[i]);}printf("\n");
}//检查容量是否够,不够进行扩容
void SLCheckCapacity(SL* ps)
{assert(ps);//满了要扩容if (ps->size == ps->capacity){//使用realloc进行扩容SLDataType* temp = (SLDataType*)realloc(ps->a, sizeof(SLDataType) * 2 * (ps->capacity));//检查是否扩容成功if (temp == NULL){perror("realloc failed");exit(-1);}ps->a = temp;ps->capacity *= 2;}
}//尾插
void SLPushBack(SL* ps, SLDataType x)
{assert(ps);//检查是否需要扩容SLCheckCapacity(ps);ps->a[ps->size] = x;ps->size++;
}//尾删
void SLPopBack(SL* ps)
{assert(ps);//暴力检查assert(ps->size > 0);ps->size--;
}//头插
void SLPushFront(SL* ps, SLDataType x)
{assert(ps);//检查是否需要扩容SLCheckCapacity(ps);//挪动数据int end = ps->size - 1;while (end >= 0){ps->a[end + 1] = ps->a[end];--end;}ps->a[0] = x;ps->size++;
}//头删
void SLPopFront(SL* ps)
{assert(ps);//判断顺序表是否为空assert(ps->size > 0);//挪动元素向前覆盖int begin = 1;while (begin < ps->size){ps->a[begin - 1] = ps->a[begin];++begin;}ps->size--;
}//查找某个值的下标
int SLFind(SL* ps, SLDataType x)
{assert(ps);for (int i = 0; i < ps->size; i++){if (ps->a[i] == x)return i;}return -1;
}//在下标为pos的位置插入x
void SLInsert(SL* ps, int pos, SLDataType x)
{assert(ps);//检查pos是否在有效范围内assert(pos >= 0 && pos <= ps->size);//检查是否需要扩容SLCheckCapacity(ps);//挪动数据int end = ps->size - 1;while (end >= pos) {ps->a[end + 1] = ps->a[end];--end;}//插入元素ps->a[pos] = x;ps->size++;
}//删除下标为pos位置的数据
void SLErase(SL* ps, int pos)
{assert(ps);//检查pos是否在有效范围内assert(pos >= 0 && pos < ps->size);//挪动元素向前覆盖int begin = pos + 1;while (begin < ps->size){ps->a[begin - 1] = ps->a[begin];++begin;}ps->size--;
}
4.3 🔔test.c 顺序表功能测试
在这里作者只给出头插头删这组测试样例,因为只需要调用前面的函数,所以就不给大家挨个测试了,下来之后大家可以自行尝试,多敲多练大家一块进步。
#include "SeqList.h"//尾插尾删检测
void TestSeqList1()
{SL s;//创建顺序表SLInit(&s);//初始化//尾插SLPushBack(&s, 1);SLPushBack(&s, 2);SLPushBack(&s, 3);SLPrint(&s);//打印//尾删SLPopBack(&s);SLPopBack(&s);SLPrint(&s);//打印//销毁顺序表SLDestroy(&s);
}int main()
{TestSeqList1();return 0;
}
📝总结
本文主要讲解:
- 线性表的定义:由n个具有相同数据类型的元素构成一个有限序列;
- 线性表的存储结构:顺序存储结构、链式存储结构;
- 顺序表的定义:用一段物理地址连续的存储单元依次存储数据元素的线性结构;
- 顺序表的分类:静态顺序表、动态顺序表;
- 顺序表的增删查改的实现。
今天的干货分享到这里就结束啦!如果觉得文章还可以的话,希望能给个三连支持一下,聆风吟的主页还有很多有趣的文章,欢迎小伙伴们前去点评,您的支持就是作者前进的最大动力!
相关文章:

【图解数据结构】顺序表实战指南:手把手教你详细实现(超详细解析)
🌈个人主页:聆风吟 🔥系列专栏:图解数据结构、算法模板 🔖少年有梦不应止于心动,更要付诸行动。 文章目录 一. ⛳️线性表1.1 🔔线性表的定义1.2 🔔线性表的存储结构 二. ⛳️顺序表…...

WordPress怎么禁用文章和页面古腾堡块编辑器?如何恢复经典小工具?
现在下载WordPress最新版来搭建网站,默认的文章和页面编辑器,以及小工具都是使用古腾堡编辑器(Gutenberg块编辑器)。虽然有很多站长说这个编辑器很好用,但是仍然有很多站长用不习惯,觉得操作太难了…...

【HarmonyOS】掌握布局组件,提升应用体验
从今天开始,博主将开设一门新的专栏用来讲解市面上比较热门的技术 “鸿蒙开发”,对于刚接触这项技术的小伙伴在学习鸿蒙开发之前,有必要先了解一下鸿蒙,从你的角度来讲,你认为什么是鸿蒙呢?它出现的意义又是…...

第4周:Pytorch——综合应用和实战项目 Day 28-30: 学习资源和社区参与
第4周:综合应用和实战项目 Day 28-30: 学习资源和社区参与 在这个阶段,我们将探索更多的学习资源并鼓励参与PyTorch和TensorFlow的社区,以进一步提升技术和融入开发者社群。 学习资源: 论文:阅读最新的机器学习和深度…...

TypeScript教程(一)在vscode中的配置TypeScript环境
TypeScript教程(一)在vscode中的配置TypeScript环境 文章目录 TypeScript教程(一)在vscode中的配置TypeScript环境一、前言二、具体步骤1、Node.js安装2、TypeScript安装3、helloworld 一、前言 未来的开发者们请上座,…...

sshpass的安装与使用
一.简介 1.定义: ssh 登陆不能在命令行中指定密码,sshpass 的出现则解决了这一问题。它允许你用 -p 参数指定明文密码,然后直接登录远程服务器,它支持密码从命令行、文件、环境变量中读取。 2.使用 sshpass 原因 使用 sshpass…...

Excel·VBA合并工作簿2
其他合并工作簿的方法,见之前的文章《ExcelVBA合并工作簿》 目录 8,合并文件夹下所有工作簿中所有工作表,按表头汇总举例 8,合并文件夹下所有工作簿中所有工作表,按表头汇总 与之前的文章《ExcelVBA合并工作簿&#x…...

linux内核原理--分页,页表,内核线性地址空间,伙伴系统,内核不连续页框分配,内核态小块内存分配器
1.分页,页表 linux启动阶段,最初运行于实模式,此阶段利用段寄存器,段内偏移,计算得到物理地址直接访问物理内存。 内核启动后期会切换到保护模式,此阶段会开启分页机制。一旦开启分页机制后,内…...

【MongoDB】下载安装、指令操作
目录 1.下载安装 2.指令 2.1.基础操作指令 2.2.增加 2.3.查询 2.4.修改 2.5.删除 前言: 关于MongoDB的核心概念请移步: 【文档数据库】ES和MongoDB的对比-CSDN博客 1.下载安装 本文以安装Windows版本的mongodb为例,Linux版本的其实…...

k8s-pvc/pv扩容记录
背景 一次聊天过程中,对方提及pvc的扩容,虽然有注意过 storageclass 有个AllowVolumeExpansion的配置(有些csi插件是不支持该配置的,比如local-volume-provisoner),但是没有实际用过,所以还是心…...

关于Unity插件TriLib使用的一点儿心得
最近做一个项目的时候,由于要求动态加载fbx或者glb等格式文件,而我们自己开发加载插件难度又有点大,所以最后使用了TriLib这个插件,现在说一点使用心得。 由于文件加载之后要对加载的内容进行复制,比如加载一个柱子&am…...

计算机二级Python基本排序题-序号45(补充)
1. 文件"singup.txt”中保存了若干条参加运动会学生的报名记录,每条记录的形式为“班级号_学号”,例如"A1_12”,将每个班级报名情按参加运动会人数从多到少排列(假设不存在人数相同的情况)并输出,…...

响应式Web开发项目教程(HTML5+CSS3+Bootstrap)第2版 例4-6 fieldset
代码 <!doctype html> <html> <head> <meta charset"utf-8"> <title>fieldset</title> </head><body> <form action"#"><fieldset><legend>学生信息</legend>姓名:&…...

html渲染优先级
在前端开发中,优先布局是指在设计和构建页面时,将页面的各个部分按照其重要性和优先级进行排序,并依次进行布局和开发。这种方法可以帮助开发团队在项目初期就确定页面结构的核心部分,从而更好地掌控项目的整体进度和优先级。且确…...

linux 更新镜像源
打开终端,备份一下旧的 源 文件,以防万一 cd /etc/apt/ ls sudo cp sources.list sources.list.bak ls然后打开清华大学开源软件镜像站 搜索一下你的linux发行版本,我这里是ubuntu发行版本 点击这个上面图中的问号 查看一下自己的版本号&a…...

【征服Redis12】redis的主从复制问题
从现在开始,我们来讨论redis集群的问题,在前面我们介绍了RDB和AOF两种同步机制,那你是否考虑过这两个机制有什么用呢?其中的一个重要作用就是为了集群同步设计的。 Redis是一个高性能的键值存储系统,广泛应用于Web应用…...

php函数 一
一 自动加载 1.1 __autoload(string $class) 类自动加载,7.2版本之后废弃。可使用sql_autoload_register()注册方法实现。 类自动加载,无返回值。 #php7.2之前function __autoload($class) {if(strpos($class, CI_) ! 0){if (file_exists(APPPATH . …...

监督学习 - 梯度提升回归(Gradient Boosting Regression)
什么是机器学习 梯度提升回归(Gradient Boosting Regression)是一种集成学习方法,用于解决回归问题。它通过迭代地训练一系列弱学习器(通常是决策树)来逐步提升模型的性能。梯度提升回归的基本思想是通过拟合前一轮模…...

【工具】使用ssh进行socket5代理
文章目录 shellssh命令详解正向代理:反向代理:本地 socks5 代理 shell ssh -D 3333 root192.168.0.11 #输入密码 #3333端口已经使用远程机进行转发设置Windows全局代理转发 socks127.0.0.1 3333如果远程机为公网ip,可通过搜索引擎查询出网…...

(delphi11最新学习资料) Object Pascal 学习笔记---第2章第六节(类型转换)
Object Pascal 学习笔记,Delphi 11 编程语言的完整介绍 作者: Marco Cantu 笔记:豆豆爸 2.6 类型转换和类型转换 正如我们所见,不能将一种数据类型的变量赋值给另一种类型的变量。原因在于,根据数据的实际表示,你…...

计算机服务器中了mallox勒索病毒怎么办,mallox勒索病毒解密数据恢复
企业的计算机服务器存储着企业重要的信息数据,为企业的生产运营提供了极大便利,但网络安全威胁随着技术的不断发展也在不断增加,近期,云天数据恢复中心接到许多企业的求助,企业的计算机服务器中了mallox勒索病毒&#…...

CPU相关专业名词介绍
CPU相关专业名词 1、CPU 中央处理器CPU(Central Processing Unit)是计算机的运算和控制核心,可以理解为PC及服务器的大脑CPU与内部存储器和输入/输出设备合称为电子计算机三大核心部件CPU的本质是一块超大规模的集成电路,主要功…...

VRRP协议负载分担
VRRP流量负载分担 VRRP负载分担与VRRP主备备份的基本原理和报文协商过程都是相同的。同样对于每一个VRRP备份组,都包含一个Master设备和若干Backup设备。与主备备份方式不同点在于:负载分担方式需要建立多个VRRP备份组,各备份组的Master设备可以不同;同一台VRRP设备可以加…...

maven 基本知识/1.17
maven ●maven是一个基于项目对象模型(pom)的项目管理工具,帮助管理人员自动化构建、测试和部署项目 ●pom是一个xml文件,包含项目的元数据,如项目的坐标(GroupId,artifactId,version )、项目的依赖关系、构建过程 ●生命周期&…...

【Java】HttpServlet类简单方法和请求显示
1、HttpServlet类简介🍀 Servlet类中常见的三个类有:☑️HttpServlet类,☑️HttpServletRequest类,☑️HttpResponse类 🐬其中,HttpServlet首先必须读取Http请求的内容。Servlet容器负责创建HttpServlet对…...

使用Rancher管理Kubernetes集群
部署前规划 整个部署包括2个部分,一是管理集群部署,二是k8s集群部署。管理集群功能主要提供web界面方式管理k8s集群。正常情况,管理集群3个节点即可,k8s集群至少3个。本文以3节点管理集群,3节点k8s集群为例 说明部署过…...

QT中操作word文档
QT中操作word文档: 参考如下内容: C(Qt) 和 Word、Excel、PDF 交互总结 Qt对word文档操作总结 QT中操作word文档 Qt/Windows桌面版提供了ActiveQt框架,用以为Qt和ActiveX提供完美结合。ActiveQt由两个模块组成: QAxContainer模…...

纯前端在线Office文档安全预览之打开Word文档后禁止打印、禁止另存为、禁止复制
在一些在线Office文档中,有很多重要的文件需要保密控制,比如:报价单、客户资料等数据,只能给公司成员查看,但是不能编辑,并且不能拷贝,打印、不能另存为,导致重要资料外泄。 可以通…...

李沐深度学习-d2lzh_pytorch模块实现
d2lzh_pytorch 模块 import random import torch import matplotlib_inline from matplotlib import pyplot as plt import torchvision import torchvision.transforms as transforms import torchvision.datasets import sys from collections import OrderedDict# --------…...

什么是OSPF?为什么需要OSPF?OSPF基础概念
什么是OSPF? 开放式最短路径优先OSPF(Open Shortest Path First)是IETF组织开发的一个基于链路状态的内部网关协议(Interior Gateway Protocol)。 目前针对IPv4协议使用的是OSPF Version 2(RFC2328&#x…...