顺序表和链表【数据结构】【基于C语言实现】【一站式速通】

目录

顺序表

顺序表的优点

顺序表的实现

1.结构体的定义

2.初始化数组

 3.插入数据

4.其余接口函数的实现

5.释放内存

顺序表的缺陷

单向链表

单向链表的优点

单向链表的实现

1.链表的定义 

2.链表的初始化

3.其余接口函数的实现

5.释放内存

单向链表的缺陷

双向链表

双向链表的优点

双向链表的实现

1.双向链表的初始化

2.链表的初始化

3.其余接口函数的实现

 4.释放内存

 双向链表的缺陷

总结


线性表(linear list)是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使

用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串...

线性表和链表的物理结构:

线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构上并不一定是连续的,

线性表在物理上存储时,通常以数组和链式结构的形式存储。画出它们的物理结构只是为了方便我们理解它们各自的特性。


顺序表

在计算机科学中,顺序表是一种常见且重要的数据结构。顾名思义,顺序表是一种按照元素在内存中的物理顺序进行存储和访问的数据结构。它可以看作是一段连续的内存空间,用于存储相同类型的元素。 

顺序表的优点

1.支持随机访问:由于顺序表在内存中是连续存储的,因此可以通过下标直接访问任何一个元素。这使得顺序表具有高效的随机访问能力,时间复杂度为O(1)。

2.有序存储:顺序表中的元素按照其在数组中的位置顺序存储,因此保持了元素的逻辑顺序。这使得顺序表适用于需要保持元素有序性的场景,例如排序、查找等操作。

3.内存紧凑:顺序表中的元素在内存中是连续存储的,不需要额外的指针来连接各个元素,因此可以更好地利用内存空间。这使得顺序表相对于链表等动态数据结构来说,具有更小的存储空间和更高的存取效率。

顺序表的实现

顺序表是用一段 物理地址连续 的存储单元依次存储数据元素的线性结构,一般情况下采用数组存
储。在数组上完成数据的增删查改。

顺序表一般都是靠数组来进行存储,如果在栈上开辟一块空间,我们需要指定数组的元素个数,如果指定个数过少,我们的数据没有办法进行存储;如果指定个数过多,又会造成资源的浪费。所以,我们选择在堆上开辟空间,这样我们可以实现一个动态存储的数据表,按需分配空间。下面我们来着手实现一个顺序表:

1.结构体的定义

因为要实现一个动态增长的版本,所以我们要给定一个数组的指针,一个记录有效数据个数的变量,和一个代表数组容量的变量,用以在我们空间不足时候进行扩容操作。

//实现一个顺序表
typedef struct Sequence
{Seqtype* a;int size;int capacity;
}SeqList;

2.初始化数组

这里要传的是结构体的指针,因为我们需要改变结构体里面的值,需要传址调用

当我们只定义而不进行初始化的时候,我们的指针会是野指针,size和capacity都会是随机数,因此我们要进行一下初始化

void SeqListInit(SeqList* ps)
{assert(ps);ps->a = NULL;ps->capacity = ps->size = 0;
}

 3.插入数据

当插入数据的时候,我们就要考虑空间够不够的问题了,经过思考其实可以发现,size是有效数据的个数,capacity是容量,当有效数据的个数刚好等于容量的时候其实就是要扩容的时候。刚开始时,我们还没有分配空间,所以我们先给定容量为4,后期空间不足再调整即可

void SeqListPushBack(SeqList* ps, Seqtype x)
{assert(ps);if (ps->size == ps->capacity){//扩容---三目操作符int newcapacity = ps->capacity > 0 ? ps->capacity * 2 : 4;Seqtype* newsapce = (Seqtype*)realloc(ps->a ,sizeof(Seqtype) * newcapacity);if (newsapce == NULL)                                                      {printf("malloc fail\n");exit(-1);}ps->a = newsapce;ps->capacity = newcapacity;}ps->a[ps->size++] = x;
}

注意:在分配空间时,我们一定要使用realloc,而不是malloc。当使用realloc时,如果给定一个空指针,那他此时就是malloc的功能。realloc与malloc的一个重要的区别就是:realloc在堆上申请空间的时候,会返回申请到的空间的指针,并把原先的内容按字节拷贝到该指针指向的数组中,而malloc不会拷贝,切记切记!!!

4.其余接口函数的实现

//接口函数
void SeqListPrint(SeqList* ps);
void SeqListInit(SeqList* ps);
void SeqListPushBack(SeqList* ps, Seqtype x);
void SeqListPushFront(SeqList* ps, Seqtype x);
void SeqListPopBack(SeqList* ps);
void SeqListDestroy(SeqList* ps);
void SeqListPopFront(SeqList* ps);
int SeqListFind(SeqList* ps, Seqtype x);
void SeqListInsert(SeqList* ps, Seqtype x, int pos);
void SeqListCheckCapacity(SeqList* ps);

这里的接口函数太多,不再一一赘述,只把几个接口函数的思想进行分析

void SeqListPrint(SeqList* ps)
{for (int i = 0; i < ps->size; i++){printf("%d ", ps->a[i]);}printf("\n");
}void SeqListCheckCapacity(SeqList* ps)
{if (ps->size == ps->capacity){//扩容int newcapacity = ps->capacity > 0 ? ps->capacity * 2 : 4;Seqtype* newsapce = (Seqtype*)realloc(ps->a, sizeof(Seqtype) * newcapacity);if (newsapce == NULL)                                                       {printf("malloc fail\n");exit(-1);}ps->a = newsapce;ps->capacity = newcapacity;}
}void SeqListPopBack(SeqList* ps)
{assert(ps);assert(ps->size > 0);ps->size--;
}void SeqListPushFront(SeqList* ps, Seqtype x)
{assert(ps);SeqListCheckCapacity(ps);int end = ps->size - 1;while (end >= 0){ps->a[end+1] = ps->a[end];end--;}ps->a[0] = x;ps->size++;
}void SeqListPopFront(SeqList* 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 SeqListFind(SeqList* ps, Seqtype x)
{assert(ps);//遍历数组int i = 0;for ( i = 0; i < ps->size; i++){if (ps->a[i] == x){break;}}return i;
}void SeqListInsert(SeqList* ps, Seqtype x, int pos)
{assert(ps);assert(pos < ps->size);SeqListCheckCapacity(ps);int end = ps->size - 1;while (end >= pos){ps->a[end + 1] = ps->a[end];end--;}ps->a[pos] = x;ps->size++;
}

我们发现在插入数据的时候,不管是头插还是为尾插,只要是插入数据,都需要进行判断空间是否充足的处理,因此我们决定把这个检查空间的功能封装成一个函数,方便我们后续的调用,这里其实就是代码复用。

void SeqListCheckCapacity(SeqList* ps)
{if (ps->size == ps->capacity){//扩容int newcapacity = ps->capacity > 0 ? ps->capacity * 2 : 4;Seqtype* newsapce = (Seqtype*)realloc(ps->a, sizeof(Seqtype) * newcapacity);if (newsapce == NULL)                                                       {printf("malloc fail\n");exit(-1);}ps->a = newsapce;ps->capacity = newcapacity;}
}

其实,在实现完Insert函数和Erase函数之后,我们就会发现,头删尾删都可以进行代码复用了,我们的代码能够得到极大的简化。大家可以试着实现一下。

5.释放内存

因为我们的数组是开辟在堆上面的,所以我们需要在使用完之后释放掉这块内存,否则就会造成内存泄漏。

void SeqListDestroy(SeqList* ps)
{assert(ps);free(ps->a);ps->a = NULL;ps->capacity = ps->size = 0;
}

顺序表的缺陷

1.空间不够了,需要扩容,扩容是有消耗的,会产生很多内存碎片。

2.头部/中间位置的插入删除,需要挪动数据,时间复杂度为O(N),挪动数据也是有消耗的

3.为避免重复扩容,一次一般都是按倍数去扩容(一般是二倍),还可能存在一定的空间浪费

4.增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。

单向链表

 针对顺序表的缺陷,设计出了链表。按需申请空间,不用了就释放空间(更加合理的使用了空间)

头部中间插入删除数据,不需要挪动数据。

链表的概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的 。

单向链表的优点

因为链表是根据顺序表的缺陷进行设计的,所以链表的优势就在于,它里面的数据不是连续存储的,而是通过一个个的指针链接起来的。因此不存在会浪费空间,造成空间碎片的问题。

1.动态性:链表的大小可以动态地进行调整,不需要事先预留固定的内存空间。在插入或删除节点时,只需要调整指针的指向,而不需要移动其他节点。这使得链表适用于频繁进行插入和删除操作的场景。

2.灵活性:相比于顺序表,链表的结构更加灵活。链表可以根据实际需求设计成单向链表、双向链表或循环链表。单向链表只有一个指针指向下一个节点,双向链表则同时具有前向和后向指针,而循环链表的尾节点指针指向头节点。根据实际需求,我们可以选择合适的链表类型。

3.内存利用率高:链表在内存中不要求连续存储,因此可以更好地利用内存空间。相比于顺序表,链表可以动态地分配内存,并且不会产生内存碎片的问题。

单向链表的实现

1.链表的定义 

我们可以根据链表的物理结构来设计链表,我们首先要定义一个变量来存储数据,因为我们还要存储下一个数据的地址,所以我们还要定义一个指针。

typedef struct Slist
{SLtype data;struct Slist* next;
}SLTnode;

2.链表的初始化

因为我们不直接把所需空间直接全部初始化完成,而是一个一个地存储,通过该结构体中指向下一个数据的指针找到下一个数据,所以链表的初始化就是先创建一个新节点,并让其中的指针指向空,防止野指针。

SLTnode* CreatListNode(SLtype x)
{//创建一个新的节点SLTnode* newnode = malloc(sizeof(SLtype));if (newnode == NULL){printf("malloc fail\n");exit(-1);}newnode->data = x;newnode->next = NULL;return newnode;
}

3.插入数据

插入数据时我们写的是二级指针,有些人可能不理解。在此进行说明:

为什么要传二级指针?

类比int类型的数据,我们在传址调用的时候,&该变量,我们在函数中用的是(int *)指针进行接受。我们如果要传址调用(int *)类型的变量,就要用二级指针(int **)来进行接收。

void SLTpushback(SLTnode** pphead, SLtype x)
{SLTnode* newnode = CreatListNode(x);if (*pphead == NULL){*pphead = newnode;}else{//找到尾节点SLTnode* tail = *pphead;while (tail->next != NULL){tail = tail->next;}//链接tail->next = newnode;}
}

3.其余接口函数的实现

void SLTprint(SLTnode* phead);
void SLTpushback(SLTnode** phead, SLtype x);
void SLTpushfront(SLTnode** phead, SLtype x);
void SLTpopback(SLTnode** phead);
void SLTpopfront(SLTnode** phead);
SLTnode* SLTfind(SLTnode* phead, SLtype x);
void SLTinsert(SLTnode** phead, SLTnode* pos , SLtype x);
void SLTDestory(SLTnode** phead);

 其余接口函数的实现不再一一赘述,只挑选重点部分进行说明

void SLTprint(SLTnode* phead)
{SLTnode* cur = phead;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("NULL\n");
}void SLTpushback(SLTnode** pphead, SLtype x)
{SLTnode* newnode = CreatListNode(x);if (*pphead == NULL){*pphead = newnode;}else{//找到尾节点SLTnode* tail = *pphead;while (tail->next != NULL){tail = tail->next;}//链接tail->next = newnode;}}void SLTpushfront(SLTnode** pphead, SLtype x)
{SLTnode* newnode = CreatListNode(x);newnode->next = *pphead;*pphead = newnode;
}void SLTpopback(SLTnode** pphead)
{assert(*pphead != NULL);if ((*pphead)->next == NULL){free(*pphead);*pphead = NULL;}else{SLTnode* tail = *pphead;SLTnode* previous = *pphead;while (tail->next){previous = tail;tail = tail->next;}free(tail);tail = NULL;previous->next = NULL;//注意:previous是一个指针,指向的是尾节点的前一个节点的地址,//我们的目的是要把该内存块中指向下一内存的地址给置空}//错误方法/*SLTnode* tail = *pphead;while (tail->next){tail = tail->next;}free(tail);tail = NULL;*/   //这种方式没办法使新链表指向原最后一个数据的指针置空,造成野指针
}void SLTpopfront(SLTnode** pphead)
{assert(*pphead != NULL);SLTnode* next = (*pphead)->next;free(*pphead);*pphead = NULL;
}
SLTnode* SLTfind(SLTnode* phead, SLtype x)
{assert(phead != NULL);//遍历链表,找到要查找的值SLTnode* tail = phead;while (tail->next){if (tail->data == x){return tail;}tail = tail->next;}return NULL;
}void SLTinsert(SLTnode** phead, SLTnode* pos, SLtype x)
{assert(*phead != NULL);SLTnode* newnode = CreatListNode( x );if (*phead == pos){SLTpushfront(phead , x);}else{SLTnode* previous = *phead;while (previous->next != pos){previous = previous->next;}previous->next = newnode;newnode->next = pos;}
}

我们在删除尾部的数据时,不能只把最后一个数据的空间free掉,还要把它的前一个数据中指向该数据的指针给置成NULL,而在单向链表中,我们可以轻松取得链表的头和尾,但是如果要访问倒数第二个值,还需要额外的指针,这也是单向链表的一个弊端!

5.释放内存

链表的节点是malloc出来的,为了防止内存泄漏,我们在使用完之后,要进行内存释放。

链表的内存释放有点特殊,因为它们不是连续存放的,开辟了多个节点,每个节点都保留着指向下一个节点的指针,所以我们要把这些节点全部free掉。

//销毁链表
void SLTDestory(SLTnode** phead)
{SLTnode* cur = *phead;while (cur != NULL){SLTnode* next = cur->next;free(cur);cur = next;}*phead = NULL;
}

单向链表的缺陷

1.不支持随机访问:在删除数据时,我们发现了,每一个数据,都要存一个指针去链接后面的数据节点,不支持随机访问(用下标直接访问第i个)【顺序表支持】

2.删除节点需谨慎:删除链表中的某个节点时,需要修改前一个节点的指针,将其指向下一个节点,然后释放被删除节点的内存。如果不仔细处理指针的更新,可能会导致内存泄漏或者链表断裂。

双向链表

双向链表的优点

 双向链表的设计可以看做是单向链表的扩展,每个节点除了存储数据外,还需要存储前继节点和后继节点的指针。这种设计使得双向链表具有以下优点:

1.可以双向遍历:由于每个节点都有前继节点和后继节点的指针,因此可以从任意一个节点开始,顺着前继节点或后继节点进行遍历。这使得双向链表在某些场景下具有比单向链表更高的遍历效率。

2.方便进行插入和删除操作:在双向链表中插入或删除节点时,只需要修改相邻节点的指针即可,不需要像单向链表那样找到前一个节点来修改指针。这使得双向链表在插入和删除操作方面更加方便。

双向链表的实现

1.双向链表的初始化

双向链表基于单向链表,只不过是又加入了一个指针,我们注意命名规范,直接定义即可。

typedef struct ListNode
{DLtype data;struct ListNode* prev;struct ListNode* next;
}DL;

2.链表的初始化

 初始化时需要注意的是,当链表中只有一个数据的时候,这时候的两个指针都指向自己。

DL* ListInit(DL* phead)
{DL* newnode = (DL*)malloc(sizeof(DL));if (newnode == NULL){printf("malloc fail\n");exit(-1);}newnode->prev = newnode;newnode->next = newnode;return newnode;
}

3.其余接口函数的实现

DL* ListInit(DL* phead);
void Listprint(DL* phead);
void Listpushfront(DL* phead, DLtype x);
void Listpushback(DL* phead, DLtype x);
void Listpopfront(DL* phead);
void Listpopback(DL* phead);
void ListInsert(DL* pos,DLtype x);
void Listpop(DL* pos);
void Listfind(DL* phead);

这里跟单向链表的逻辑其实相差不多,只是过程比较繁琐,在这里我就挑选其中的几个进行实现 

void Listprint(DL* phead)
{assert(phead);DL* cur = phead->next;while (cur != phead){printf("%d ", cur->data);cur = cur->next;}printf("\n");
}void Listpushfront(DL* phead, DLtype x)
{ListInsert(phead->next, x);
}void Listpushback(DL* phead, DLtype x)
{ListInsert(phead, x);
}void ListInsert(DL* pos, DLtype x)
{DL* newnode = (DL*)malloc(sizeof(DL));if (newnode == NULL){printf("malloc fail\n");exit(-1);}newnode->data = x;DL* prev = pos->prev;if (prev == NULL) // 如果pos是头节点{newnode->next = pos;pos->prev = newnode;}else // 正常情况{prev->next = newnode;newnode->prev = prev;newnode->next = pos;pos->prev = newnode;}
}

 4.释放内存

我们在释放current的内存之前保存下一个空间的地址,然后cur一直往后面走,free掉经过的空间,当cur指向NULL的时候,所有的空间都被free完了 

void ListFree(DL** phead)
{DL* current = phead;while (current != NULL){DL* next = current->next;free(current);current = next;}*phead = NULL;
}

 双向链表的缺陷

1.需要更多的存储空间:相比于单向链表,双向链表需要多存储一个指针,因此需要更多的存储空间。这对于需要大量使用链表的应用程序来说可能会成为内存限制的瓶颈。

2.实现较为繁琐:相比于单向链表,双向链表需要同时操作两个指针,很容易把人绕晕。


总结

通过上面的分析,我们发现不管是线性表还是单向链表,甚至是双向链表,都有自己的优缺点,我们要根据实际的使用场景来选择要使用哪一种方式来进行存储数据,快捷、高效地处理问题。

今天的分享到这里就结束了,欢迎讨论交流~

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ELK日志解决方案

ELK日志解决方案 ELK套件日志系统应该是Elasticsearch使用最广泛的场景之一了&#xff0c;Elasticsearch支持海量数据的存储和查询&#xff0c;特别适合日志搜索场景。广泛使用的ELK套件(Elasticsearch、Logstash、Kibana)是日志系统最经典的案例&#xff0c;使用Logstash和Be…

CVPR——Latex模版下载

CVPR官网 -> AuthorGuidelines 链接&#xff1a;AuthorGuidelines

怎么把几百M大小的视频做成二维码?扫码播放视频在线教程

怎么把几百M大小的视频做成一个二维码展示呢&#xff1f;通过二维码来作为视频的载体是现在很常用的一种手段&#xff0c;通过这种方式不仅成本比较低&#xff0c;而且传播速度也比较快&#xff0c;通过访问云端数据就可以播放视频。 视频二维码生成的方法一般会通过二维码生成…

微信小程序(二十)Vant组件库的配置

教程很详细&#xff0c;直接上过程 上一篇 官方文档也有&#xff0c;但是因为版本的更新&#xff0c;官方文档并没有跟着改变&#xff0c;这里我写一份最新版能用的教程 &#xff08;口头禅还是不能少的&#x1f923;&#x1f923;&#x1f923;&#xff09; 灵魂拷问&#xf…

【RT-DETR有效改进】反向残差块网络EMO | 一种轻量级的CNN架构(轻量化网络,参数量下降约700W)

👑欢迎大家订阅本专栏,一起学习RT-DETR👑 一、本文介绍 本文给大家带来的改进机制是反向残差块网络EMO,其的构成块iRMB在之前我已经发过了,同时进行了二次创新,本文的网络就是由iRMB组成的网络EMO,所以我们二次创新之后的iEMA也可以用于这个网络中,再次形成二次…

Kotlin快速入门5

Kotlin的继承与重写 kotlin的继承 Kotlin中所有类都继承自Any类&#xff0c;Any类是所有类的超类&#xff0c;对于没有超类型声明的类是默认超类&#xff08;Any 不是 java.lang.Object&#xff09;&#xff1a; class LearnKotlin // 默认继承自Any Any类默认提供三个函数…

Android 13.0 SystemUI下拉状态栏定制二 锁屏页面横竖屏时钟都居中功能实现二

1.前言 在13.0的系统rom定制化开发中,在关于systemui的锁屏页面功能定制中,由于在平板横屏锁屏功能中,时钟显示的很大,并且是在左旁边居中显示的, 由于需要和竖屏显示一样,所以就需要用到小时钟显示,然后同样需要居中,所以就来分析下相关的源码,来实现具体的功能 如图…

elementplus Dialog 对话框设置距离页面顶部的距离

默认为 15vh&#xff0c;当弹窗过于高的时候&#xff0c;这个距离其实是不合适的 <el-dialogv-model"dialogVisible"title"Tips"width"30%":before-close"handleClose"top"6vh"><span>This is a message</s…

KMP板子 前缀跳后缀

目录 原理&#xff1a; 板子&#xff1a; 原理&#xff1a; 出现重复【 存在部分前缀等于后缀 &#xff08;自己的前面一部分跟后面一部分一样的&#xff09; 】的时候&#xff0c;可以跳&#xff01; 来源&#xff1a;KMP算法中next数组的理解 - 知乎 (zhihu.com) &#xf…

JVM 笔记

JVM HotSpot Java二进制字节码的运行环境 好处&#xff1a; 一次编写&#xff0c;到处运行自动内存管理&#xff0c;具有垃圾回收功能数组下标越界检查多态&#xff08;虚方法表&#xff09; JVM组成 类加载子系统&#xff08;Java代码转换为字节码&#xff09;运行时数据…

Qt6入门教程 12:QAbstractButton

目录 一.状态 二.信号 三.使用 1.自定义按钮 2.多选 3.互斥 QAbstractButton类实现了一个抽象按钮&#xff0c;并且让它的子类来指定如何处理用户的动作&#xff0c;并指定如何绘制按钮。QAbstractButton类是所有按钮控件的基类。 QAbstractButton提供…

Python笔记14-实战小游戏飞机大战(上)

文章目录 功能规划安装pygame绘制游戏窗口添加玩家飞机图像屏幕上绘制飞船代码重构驾驶飞船全屏模式射击 本示例源码地址 点击下载 功能规划 玩家控制一艘最初出现在屏幕底部中央的飞船。玩家可以使用箭头键左右移动飞船&#xff0c;还可使用空格键射击。游戏开始时&#xff…

1948-2022年金融许可信息明细数据

1948-2022年金融许可信息明细数据 1、时间&#xff1a;1948-2022年 2、来源&#xff1a;银监会&#xff08;银监会许可证发布系统&#xff09; 3、指标&#xff1a;来源表、机构编码、机构名称、所属银行、机构类型、业务范围、机构住所、地理坐标、行政区划代码、所属区县、…

Leetcode—2942. 查找包含给定字符的单词【简单】

2023每日刷题&#xff08;一零一&#xff09; Leetcode—2942. 查找包含给定字符的单词 实现代码 class Solution { public:vector<int> findWordsContaining(vector<string>& words, char x) {vector<int> ans;for(int i 0; i < words.size(); i)…

【Linux】动态库和静态库——动态库和静态库的打包和使用、gcc编译、拷贝到系统默认的路径、建立软连接

文章目录 动态库和静态库1.静态库和动态库的介绍2.静态库的打包和使用2.1生成静态库2.2使用静态库的三种方式2.2.1gcc编译2.2.2拷贝到系统默认的路径2.2.3建立软连接 3.动态库的打包和使用3.1生成动态库3.2使用动态库3.3解决加载不到动态库的方法 动态库和静态库 1.静态库和动…

linux 下scrcpy 手机投屏到电脑,QT+ffmpeg 获取视频流,处理等等

linux 下scrcpy 手机投屏到电脑,QT+ffmpeg 获取视频流,处理 1 安装 scrcpy 地址 https://github.com/Genymobile/scrcpy 转到 relese 下载 我这里下载的是linux系统 v2.3.1 版本 scrcpy-2.3.1.tar.gz 下载 scrcpy-server v2.3.1 版本 scrcpy-server-v2.3.1 解压scrcpy-2.3…

林浩然的奇幻编程之旅:幽默探索图灵的智慧宇宙

林浩然的奇幻编程之旅&#xff1a;幽默探索图灵的智慧宇宙 Lin Haoran’s Fantastical Coding Journey: Humorous Exploration of Turing’s Intelligent Universe 林浩然&#xff0c;一个对计算机科学怀揣着无尽好奇的探索者&#xff0c;像一只闯入魔法森林的小狐狸&#xff0…

网安培训第二期——sql注入+中间件+工具

文章目录 宽字节注入插入注入二次注入PDO模式(动态靶机&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;)sql注入读取文件sql注入导出文件linux命令 10.12笔记sqlmapsqlmap参数 10.13笔记sqlmap 文件读写前后缀常用tamper及适用场景 10.…

Excel·VBA时间范围筛选及批量删除整行

看到一个帖子《excel吧-筛选开始时间&#xff0c;结束时间范围内的所有记录》&#xff0c;根据条件表中的开始时间和结束时间构成的时间范围&#xff0c;对数据表中的开始时间和结束时间范围内的数据进行筛选 目录 批量删除整行&#xff0c;整体删除批量删除整行&#xff0c;分…

多维时序 | Matlab实现DBO-LSTM蜣螂算法优化长短期记忆神经网络多变量时间序列预测

多维时序 | Matlab实现DBO-LSTM蜣螂算法优化长短期记忆神经网络多变量时间序列预测 目录 多维时序 | Matlab实现DBO-LSTM蜣螂算法优化长短期记忆神经网络多变量时间序列预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现DBO-LSTM多变量时间序列预测&#x…