C语言数据结构-链表

news2025/1/11 18:38:30

C语言数据结构-链表

  • 1.单链表
    • 1.1概念与结构
    • 1.2结点
    • 3.2 链表性质
    • 1.3链表的打印
    • 1.4实现单链表
      • 1.4.1 插入
      • 1.4.2删除
      • 1.4.3查找
      • 1.4.4在指定位置之前插入或删除
      • 1.4.5在指定位置之后插入或删除
      • 1.4.6删除指定位置
      • 1.4.7销毁链表
  • 2.链表的分类
  • 3.双向链表
    • 3.1实现双向链表
      • 3.1.1尾插
      • 3.1.2头插
      • 3.1.3打印
      • 3.1.4尾删
      • 3.1.5头删
      • 3.1.6找到指定位置
      • 3.1.7在指定位置pos之后插入数据
      • 3.1.8删除指定位置的数据
      • 3.1.9销毁链表

1.单链表

1.1概念与结构

数据结构是存储并管理数据。

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

1.2结点

与顺序表不同的是,链表里的每节“车厢”都是独立申请下来的空间,我们称之为“结点”。

结点的组成主要有两个部分:当前结点要保存的数据和保存下一个结点的地址(指针变量)。

链表结点的组成部分:要存储的数据+保存下一个结点地址的指针。
在这里插入图片描述

图中指针变量plist保存的是第一个结点的地址,我们称plist此时“指向”第一个结点,如果我们希望plist“指向”第二个结点时,只需要修改plist保存的内容为0x0012FFA0.

链表中每个结点都是独立申请的(即需要插入数据时才去申请一块结点的空间),我们需要通过指针变量来保存下一个结点位置才能从当前结点找到下一结点。

指向下一个结点的地址,就可以找到下一个结点。

在这里插入图片描述
假设当前保存的结点为整型:

typedef int SLTDateType;
typedef struct SListNode
{
	SLTDateType data;//结点数据
	struct SListNode* next;//指针变量用保存下一个结点的地址,用的是指针,指向下一个节点的指针,下一个结点的类型也是结构体
}SListNode;

3.2 链表性质

1.链式机构在逻辑上是连续的,在物理结构上不一定连续
2.结点一般是从堆上申请的
3/从堆上申请来的空间,是按照一定策略分配出来的,每次申请的空间可能连续,可能不连续。

数组:数据与数据之间的地址是连续的。
链表:数据与数据之间的地址不一定是连续的。

链表也是线性表的一种。

逻辑结构:一定是线性的。
物理结构:不一定是线性的。

结合前面学到的结构体知识,我们可以给出每个结点对应的结构体代码。

当我们想要保存一个整型数据时,实际是向操作系统申请了一块内存,这个内存不仅要保存整型数据,也需要保存下一个结点的地址(当下一个结点为空时保存的地址为空)。

当我们想要从第一个结点走到最后一个结点时,只需要在当前结点拿上下一个结点的地址就可以了。

1.3链表的打印

链表实现从头到尾打印:

listnode.c:

// 单链表打印
void SListPrint(SListNode* plist)
{
	SListNode* pcur = plist;
	while (pcur != NULL)
	{
		printf("%d-> ", pcur->data);
		pcur = pcur->next;/将pcur移动到下一个结点
		//把下一个节点的地址给了pcur,就相当于移动到下一个结点了。
	}
	printf("NULL\n");

在这里直接使用plist来遍历结果也是一样的,重新创建个pcur指针,是为了避免由于指针指向的改变,导致无法重新找到链表的首结点。

链表同样也有增删改查等操作,接下来我们来实现单链表的头插和尾插

1.4实现单链表

1.4.1 插入

要插入,就要向操作系统申请结点大小的空间,存储要插入的数据。不论是头插还是尾插,都需要向操作系统申请结点大小的一块空间,所有将向操作动态申请一块空间,分装成一个函数:

// 动态申请一个节点
SListNode* BuySListNode(SLTDateType x)
{
	SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
	if (newnode == NULL)
	{
		perror("malloc fail!\n");
		exit(1);
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}

需要注意的是:
传值:形参的改变不会影响实参
传地址:形参的改变影响实参

接受一级指针的地址用二级指针
在这里插入图片描述

// 单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x)
{
	assert(pplist);
	SListNode* newnode = BuySListNode(x);
	if (*pplist == NULL)
	{
	//如果pplist为空,那么申请的结点就为首节点,直接将首节点指向*pplist
		//链表为空
		*pplist = newnode;
	}
	else
	{
		//非空
		SListNode* ptail = *pplist;//防止找不到头节点
		首先我们要先找到该链表的尾节点。
		while (ptail->next)
		{
			ptail = ptail->next;//相当于遍历,挪过去了 .
		}
		ptail->next = newnode;
	}
}

时间复杂度为:O(n)

如果只传一级指针,函数内部修改这个指针并不会影响到函数外部的指针;传二级指针就不同了。二级指针指向的是头指针的地址。在函数内部通过二级指针修改头指针的值,就可以真正改变头指针的值,使得在函数外部也可以看到链表头指针的更新。

在这里插入图片描述

// 单链表的头插
void SListPushFront(SListNode** pplist, SLTDateType x)
{
	assert(pplist);
	SListNode* newnode = BuySListNode(x);//申请一个节点.
	newnode->next = *pplist;//指向第一个节点的指针
	*pplist = newnode;//现在的头节点是nednode了,将pplist指向newnode。
}

时间复杂度为:O(1)

1.4.2删除

// 单链表的尾删
void SListPopBack(SListNode** pplist)
{
	assert(pplist&&*pplist);//链表不能为空,就是指向链表第一个结点的指针不能为空
	//要删除尾节点,就需要找到尾节点,需要遍历。
	if ((*pplist)->next == NULL)
	{
		free(*pplist);
		*pplist = NULL;
	}
	else {
	//需要定义一个游标,将尾节点的前一个结点保存下来,然后直接将NULL指向游标,就是删除了。
		SListNode* pcur = NULL;
		SListNode* ptail = *pplist;
		while (ptail->next)
		{
		//赋值
			pcur = ptail;
			ptail = ptail->next;
		}
		pcur->next = NULL;
		free(ptail);
		ptail = NULL;
	}
}
// 单链表头删
void SListPopFront(SListNode** pplist)//** pplist是第一个结点
{
	assert(pplist && *pplist);//传过来的参数不能为空,链表不能为空
	SListNode* next = (*pplist)->next;//先将下一个结点保存下来。
	free(*pplist);//释放掉原来的头节点
	*pplist = next;
}

1.4.3查找

// 单链表查找
SListNode* SListFind(SListNode* plist, SLTDateType x)//不要求形参发生改变,所以传一级指针就可以了
{
	SListNode* pcur = plist;
	while (pcur)
	{
		if (pcur->data == x)
		{
			return pcur;
		}
		pcur=pcur->next;//相当于++
	}
	return NULL;//未找到
}

1.4.4在指定位置之前插入或删除

在指定位置之前插入会改变pos之前和pos位置的结点,所以需要将po之前的结点保存下来。

// 在pos的前面插入
void SLTInsert(SListNode** pplist, SListNode* pos, SLTDateType x)//需要改变形参,传入二级指针
{
	assert(pos&&pplist);
	if (pos == *pplist)//头插
	{
		SListPushFront(pplist, x);
	}
	else {
		SListNode* newnode = BuySListNode(x);
		SListNode* prev = *pplist;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		newnode->next = pos;
		prev->next = newnode;
	}
}

在这里插入图片描述

1.4.5在指定位置之后插入或删除

//插入
// 单链表在pos位置之后插入x
// 分析思考为什么不在pos位置之前插入?
void SListInsertAfter(SListNode* pos, SLTDateType x)//不需要知道头结点也可以。
{
	assert(pos);
	SListNode* newnode = BuySListNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
}
//删除
// 单链表删除pos位置之后的值
// 分析思考为什么不删除pos位置?
void SListEraseAfter(SListNode* pos)
{
	assert(pos&&pos->next);
	SListNode* pre = pos->next;
	pos->next = pre->next;
	free(pre);
	pre = NULL;
}

1.4.6删除指定位置

// 删除pos位置
void SLTErase(SListNode** pplist, SListNode* pos)
{
	assert(pos&&pplist);
	//头删
	if (pos==*pplist)
	{
		SListPopFront(pplist);
	}
	else {
		SListNode* pre = *pplist;//保存头结点
		while (pre->next != pos)//向后遍历
		{
			pre = pre->next;
		}
		pre->next = pos->next;//将pre->next指向pos->next,就将pos删除了。
		free(pos);//释放pos的内存
		pos = NULL;
	}
}

在这里插入图片描述

1.4.7销毁链表

//销毁链表
void SLTDestroy(SListNode** pphead)
{
	SListNode* pcur = *pphead;
	while (pcur)
	{
		SListNode* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	*pphead = NULL;
}

2.链表的分类

链表的结构非常多样:
在这里插入图片描述
链表的说明:
在这里插入图片描述
①单向或双向:
单向是只能从左往右遍历;
双向是既能从左往右遍历,也能从右往左遍历。

双向链表有三个,指向前一个结点的指针(前驱结点),指向后一个结点的指针(后继结点)和数据域。

②带头或者不带头:
在前面说的单链表中的“头结点”,该头结点是链表的首节点(第一个结点),实际这样的称呼是错误的,因为链表中存在一种链表叫做带头链表(不是指链表里第一个有效的结点),这里的头结点指的是,哨兵位(不保存任何有效数据,只是占位置的)。

如带头链表中,只有头结点,那么我们就称该链表为空。

带头链表的意义就是:占位置,不需要判断链表的头是否为空。

③循环或者不循环
循环链表的尾结点不为空,指向第一个有效的结点。

虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:单链表和双向带头循环链表

1.无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等。
2.带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构带来很多优势,实现反而简单了。

3.双向链表

在这里插入图片描述

3.1实现双向链表

结点由三个部分组成:
前驱结点、后驱结点和数据。

3.1.1尾插

在这里插入图片描述

//尾插
void LTPushBack(LTNode* phead, LTDatatype x)
{
	//为什么这里传的是一级指针?
	//pphead不会发生改变,哨兵位的地址不会改变。
	//不需要传地址过去,一级指针
	//如果哨兵位发生改变,就传二级指针
	assert(phead);
	LTNode* newnode = BuyListNode(x);
	//phead phead->prev newnode
	newnode->prev = phead->prev;
	newnode->next = phead;
	phead->prev->next = newnode;
	phead->prev = newnode;
}

3.1.2头插

在这里插入图片描述

//头插
void LTPushFront(LTNode* phead, LTDatatype x)
{
	assert(phead);
	//phead phead->next newnode
	LTNode* newnode = BuyListNode(x);
	//先改变d1

	newnode->prev = phead;
	newnode->next = phead->next;

	phead->next->prev = newnode;
	phead->next = newnode;
}

3.1.3打印

//打印
void LTPrint(LTNode* phead)
{
	LTNode* pcur = phead->next;
	//不等于哨兵位就继续向下遍历
	while (pcur != phead)
	{
		printf("%d ->", pcur->data);
		pcur = pcur->next;
	}
	printf("\n");
}

3.1.4尾删

首先需要判断链表是否为空:

//判断是否为空
bool LTEmpty(LTNode* phead)
{
	assert(phead);
	//判断哨兵位的下一个是不是指向它自己
	//如果指向它自己说明链表为空
	return phead->next == phead;
	//如果return返回的是真,传给下面函数的是真
}

在这里插入图片描述

//尾删
void LTPopBack(LTNode* phead)
{//传过来的是真,就是phead->next==phead,就说明已经遍历结束,给前面加!表示已经为空。
	assert(!LTEmpty(phead));
	//判断是否为空
	LTNode* del = phead->prev;
	//phead del->prev del
	del->prev->next = phead;
	phead->prev = del->prev;
	free(del);
	del = NULL;
}

3.1.5头删

在这里插入图片描述

//头删
void LTPopFront(LTNode* phead)
{
//
	assert(!LTEmpty(phead));
	LTNode* del = phead->next;
	del->next->prev = phead;
	phead->next = del->next;
	free(del);
	del = NULL;
}

3.1.6找到指定位置

//找到指定位置
LTNode* LTFind(LTNode* phead, LTDatatype x)
{
	//需要遍历
	LTNode* pcur = phead->next;
	while (pcur != phead)
	{
		if (pcur->data == x)
		{
			return pcur;
		}
		pcur = pcur->next;
	}
	return NULL;
}

3.1.7在指定位置pos之后插入数据

在这里插入图片描述

//在指定位置pos之后插入数据
void LTInsert(LTNode* pos, LTDatatype x)
{
	assert(pos);
	LTNode* newnode = BuyListNode(x);
	newnode->next = pos->next;
	newnode->prev = pos;
	pos->next->prev = newnode;
	pos->next = newnode;
}

3.1.8删除指定位置的数据

//在指定位置pos删除数据
void LTErase(LTNode* pos)
{
	assert(pos);
	pos->prev->next = pos->next;
	pos->next->prev = pos->prev;
	free(pos);
	pos = NULL;
}

3.1.9销毁链表

//销毁
void LTDesTroy(LTNode** pphead)
{
	LTNode* pcur = (*pphead)->next;
	while (pcur != *pphead)
	{
		//释放之前把pcur的下一个结点先保存起来
		LTNode* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	free(*pphead);
	*pphead = NULL;
}

#4.源代码

list.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>

typedef int LTDatatype;

typedef struct ListNode {
	LTDatatype data;
	struct ListNode* next;
	struct ListNode* prev;
}LTNode;

//申请一个结点
LTNode* BuyListNode(LTDatatype x);
//初始化
LTNode* LTInit();//调用
//void LTinit(LTNode** pphead);

//销尽量也传一级,这是保持接口的一致性。
//销毁
//void LTDesTroy(LTNode**phead);//要将哨兵位也销毁掉,所以要传二级指针
void LTDesTroy(LTNode* phead);
//打印
void LTPrint(LTNode* phead);

//判断是否为空
bool LTEmpty(LTNode* phead);
//尾插
void LTPushBack(LTNode* phead, LTDatatype x);
//头插
void LTPushFront(LTNode* phead, LTDatatype x);


//尾删
void LTPopBack(LTNode* phead);
//头删
void LTPopFront(LTNode* phead);

//在指定位置pos之后插入数据
void LTInsert(LTNode* pos, LTDatatype x);

//在指定位置pos删除数据
void LTErase(LTNode* pos);

//找到指定位置
LTNode* LTFind(LTNode* phead, LTDatatype x);

list.c

#define _CRT_SECURE_NO_WARNINGS 1

#include"list.h"

//初始化
LTNode* LTInit()
{
	//创建头结点
	//LTNode* phead = (LTNode*)malloc(sizeof(LTNode));
	//phead->data = -1;
	//phead->next = phead->prev;//需要将头结点指向它自己,就成为了带头双向循环链表
	//phead->prev = phead;
	LTNode* phead = BuyListNode(-1);
	return phead;//通过返回的方式
}

//销毁
//void LTDesTroy(LTNode** pphead)
//{
//	LTNode* pcur = (*pphead)->next;
//	while (pcur != *pphead)
//	{
//		//释放之前把pcur的下一个结点先保存起来
//		LTNode* next = pcur->next;
//		free(pcur);
//		pcur = next;
//	}
//	free(*pphead);
//	*pphead = NULL;
//}
void LTDesTroy(LTNode* phead)
{
	LTNode* pcur = phead->next;
	while (pcur != phead)
	{
		LTNode* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	free(phead);
	phead = NULL;
}
//我把我的博客链接给你,你给我写。
//void LTinit(LTNode** pphead)
//{
//	assert(pphead);
//	*pphead = (LTNode*)malloc(sizeof(LTNode));
//	(*pphead)->data = -1;
//	(*pphead)->next = (*pphead)->prev = *pphead;
//
//}

//创建一个新的结点
LTNode* BuyListNode(LTDatatype x)
{
	LTNode* node = (LTNode*)malloc(sizeof(LTNode));
	if (node == NULL)
	{
		perror("malloc fail!\n");
		exit(1);
	}
	node->data = x;
	node->next = node->prev = node;
	return node;
}

//打印
void LTPrint(LTNode* phead)
{
	LTNode* pcur = phead->next;
	//不等于哨兵位就继续向下遍历
	while (pcur != phead)
	{
		printf("%d ->", pcur->data);
		pcur = pcur->next;
	}
	printf("\n");
}

//判断是否为空
bool LTEmpty(LTNode* phead)
{
	assert(phead);
	//判断哨兵位的下一个是不是指向它自己
	//如果指向它自己说明链表为空
	return phead->next == phead;
}

//尾插
void LTPushBack(LTNode* phead, LTDatatype x)
{
	//为什么这里传的是一级指针?
	//pphead不会发生改变,哨兵位的地址不会改变。
	//不需要传地址过去,一级指针
	//如果哨兵位发生改变,就传二级指针
	assert(phead);
	LTNode* newnode = BuyListNode(x);
	//phead phead->prev newnode
	newnode->prev = phead->prev;
	newnode->next = phead;
	
	phead->prev->next = newnode;
	phead->prev = newnode;
}

//头插
void LTPushFront(LTNode* phead, LTDatatype x)
{
	assert(phead);
	//phead phead->next newnode
	LTNode* newnode = BuyListNode(x);
	//先改变d1

	newnode->prev = phead;
	newnode->next = phead->next;

	phead->next->prev = newnode;
	phead->next = newnode;
}

//尾删
void LTPopBack(LTNode* phead)
{//不用判断是否为空
	assert(!LTEmpty(phead));
	//判断是否为空
	LTNode* del = phead->prev;
	//phead del->prev del
	del->prev->next = phead;
	phead->prev = del->prev;
	free(del);
	del = NULL;
}

//头删
void LTPopFront(LTNode* phead)
{
	assert(!LTEmpty(phead));
	LTNode* del = phead->next;
	del->next->prev = phead;
	phead->next = del->next;
	free(del);
	del = NULL;
}

//在指定位置pos之后插入数据
void LTInsert(LTNode* pos, LTDatatype x)
{
	assert(pos);
	LTNode* newnode = BuyListNode(x);
	newnode->next = pos->next;
	newnode->prev = pos;
	pos->next->prev = newnode;
	pos->next = newnode;
}

//找到指定位置
LTNode* LTFind(LTNode* phead, LTDatatype x)
{
	//需要遍历
	LTNode* pcur = phead->next;
	while (pcur != phead)
	{
		if (pcur->data == x)
		{
			return pcur;
		}
		pcur = pcur->next;
	}
	return NULL;
}

//在指定位置pos删除数据
void LTErase(LTNode* pos)
{
	assert(pos);
	pos->prev->next = pos->next;
	pos->next->prev = pos->prev;
	free(pos);
	pos = NULL;
}

test.c

#define _CRT_SECURE_NO_WARNINGS 1

#include"list.h"

void test()
{
	LTNode* plist = LTInit();//创建一个双向带头循环链表。
	/*LTNode* plist = NULL;
	LTInit(&plist);*///传一级指针的地址,实参的改变会影响实参。
	//上面两种方式都可以。
	LTPushBack(plist, 1);
	LTPushBack(plist, 2);
	LTPushBack(plist, 3);
	LTPushBack(plist, 4);
	LTPushBack(plist, 5);
	LTPrint(plist);
	//LTPopFront(plist);
	//LTPrint(plist);
	//LTNode* find = LTFind(plist, 2);
	//if (find == NULL)
	//{
	//	printf("未找到!\n");
	//}
	//else {
	//	printf("找到了!\n");
	//}
	//LTInsert(find, 99);
	//LTPrint(plist);
	//LTErase(find);
	//LTPrint(plist);
	LTDesTroy(plist);
	plist = NULL;//传一级需要手动将plist置为空。
}
int main()
{
	test();
	return 0;
}

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

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

相关文章

计算机网络 网络安全基础——针对实习面试

目录 网络安全基础你了解被动攻击吗&#xff1f;你了解主动攻击吗&#xff1f;你了解病毒吗&#xff1f;说说基本的防护措施和安全策略&#xff1f; 网络安全基础 网络安全威胁是指任何可能对网络系统造成损害的行为或事件。这些威胁可以是被动的&#xff0c;也可以是主动的。…

上海乐鑫科技一级代理商飞睿科技,ESP32-C61高性价比WiFi6芯片高性能、大容量

在当今快速发展的物联网市场中&#xff0c;无线连接技术的不断进步对智能设备的性能和能效提出了更高要求。为了满足这一需求&#xff0c;乐鑫科技推出了ESP32-C61——一款高性价比的Wi-Fi 6芯片&#xff0c;旨在为用户设备提供更出色的物联网性能&#xff0c;并满足智能设备连…

初识java(2)

大家好&#xff0c;今天我们来讲讲java中的数据类型。 java跟我们的c语言的数据类型有一些差别&#xff0c;那么接下来我们就来看看。 一.字面常量&#xff0c;其中&#xff1a;199&#xff0c;3.14&#xff0c;‘a’&#xff0c;true都是常量将其称为字面常量。&#xff08;…

MMCM DRP动态配置方法(超详细讲解)

一、MMCM 源语介绍 1、调用源语 2、调用Clocking Wizard IP 调用Clocking Wizard IP核选择使用MMCM资源时&#xff0c;IP内部也是调用的MMCM源语。 Clocking Wizard IP中启用MMCM DRP接口方法&#xff1a; 在Clocking Wizard IP中设置分频倍频系数方法&#xff1a; IP核中生…

对于GC方面,在使用Elasticsearch时要注意什么?

大家好&#xff0c;我是锋哥。今天分享关于【对于GC方面&#xff0c;在使用Elasticsearch时要注意什么&#xff1f;】面试题。希望对大家有帮助&#xff1b; 对于GC方面&#xff0c;在使用Elasticsearch时要注意什么&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java…

Spring Boot 与 Spring Cloud Alibaba 版本兼容对照

版本选择要点 Spring Boot 3.x 与 Spring Cloud Alibaba 2022.0.x Spring Boot 3.x 基于 Jakarta EE&#xff0c;javax.* 更换为 jakarta.*。 需要使用 Spring Cloud 2022.0.x 和 Spring Cloud Alibaba 2022.0.x。 Alibaba 2022.0.x 对 Spring Boot 3.x 的支持在其发行说明中…

在Vue3项目中引入省市区联动插件

1. 打开HBuilder X 图1 2. 新建一个空项目 文件->新建->项目->uni-app 填写项目名称&#xff1a;vue3demo 选择项目存放目录&#xff1a;D:/HBuilderProjects 一定要注意vue的版本&#xff0c;当前选择的版本为vue3 图2 点击“创建”之后进入项目界面 图3 其中各文件…

STM32C011开发(3)----Flash操作

STM32C011开发----3.Flash操作 概述硬件准备视频教学样品申请源码下载参考程序生成STM32CUBEMX串口配置堆栈设置串口重定向FLASH数据初始化FLASH 读写演示 概述 STM32C011 系列微控制器内置 Flash 存储器&#xff0c;支持程序存储与数据保存&#xff0c;具备页面擦除、双字写入…

JVM详解:垃圾回收机制

java作为大型服务开发的主流语言&#xff0c;其运行会占用大量的内存空间&#xff0c;那么合理的使用有限的服务器资源至关重要。和大多数翻译性语言一样&#xff0c;java的运行环境jvm也内置垃圾回收机制&#xff0c;其通过一些合理的算法组合&#xff0c;定时来对堆中保存的不…

【拥抱AI】如何查看Milvus的使用情况?

查看Milvus的使用情况和性能指标可以帮助你了解数据库的健康状况、性能指标和资源使用情况。以下是一些常用的方法和工具&#xff0c;帮助你全面监控和查看Milvus的使用情况和性能指标。 1. 查看日志 Milvus的日志文件记录了运行时的各种信息&#xff0c;包括错误、警告和调…

基于Netty实现聊天室

前言 了解了Netty的基本功能和相关概念&#xff0c;使用基于Netty实现多人聊天的功能。 需求 1.服务端能够接收客户端的注册&#xff0c;并且接受用户的信息注册 2.服务端能够处理客户端发送的消息&#xff0c;并且根据消息类型进行私发或者广播发送消 3.服务端能够私发消…

利用 Jsoup 进行高效 Web 抓取与 HTML 处理

Jsoup 是一款 Java 的 HTML 解析器&#xff0c;可直接解析某个 URL 地址、HTML 文本内容。它提供了一套非常省力的 API&#xff0c;可通过 DOM&#xff0c;CSS 以及类似于 JQuery 的操作方法来取出和操作数据。 官网&#xff1a;https://jsoup.org/ 中文文档&#xff1a;Jsou…

【c语言】文件操作详解 - 从打开到关闭

文章目录 1. 为什么使用文件&#xff1f;2. 什么是文件&#xff1f;3. 如何标识文件&#xff1f;4. 二进制文件和文本文件&#xff1f;5. 文件的打开和关闭5.1 流和标准流5.1.1 流5.1.2 标准流 5.2 文件指针5.3 文件的打开和关闭 6. 文件的读写顺序6.1 顺序读写函数6.2 对比一组…

004 逻辑变量与运算

当0和1表示逻辑状态时&#xff0c;两个二进制数码按照某种特定的因果关系进行的运算——就叫&#xff1a;逻辑运算 1.二值逻辑变量与基本逻辑运算 逻辑代数: 与普通代数不同,逻辑代数中的变量只有0和1两个可取值&#xff0c;它们分别用来表示完全两个对立的逻辑状态 逻辑运…

Deepnote、JupyterLab、Google Colab、Amazon SageMaker、VS Code对比

功能比较 平台语言支持扩展性数据连接可视化能力DeepnotePython、R、SQL中等&#xff0c;依赖云端支持主要云平台&#xff08;BigQuery、Snowflake等&#xff09;内置仪表盘与交互图表JupyterLab多种语言&#xff0c;插件支持广泛极高&#xff0c;完全可自定义使用库&#xff…

网络安全中的数据科学如何重新定义安全实践?

组织每天处理大量数据&#xff0c;这些数据由各个团队和部门管理。这使得全面了解潜在威胁变得非常困难&#xff0c;常常导致疏忽。以前&#xff0c;公司依靠 FUD 方法&#xff08;恐惧、不确定性和怀疑&#xff09;来识别潜在攻击。然而&#xff0c;将数据科学集成到网络安全中…

C语言数据结构与算法--简单实现队列的入队和出队

&#xff08;一&#xff09;队列的基本概念 和栈相反&#xff0c;队列(Queue)是一种先进先出&#xff08;First In First Out&#xff09;的线性表。只 允许在表的一端进行插入&#xff0c;而在另一端删除元素&#xff0c;如日常生活中的排队现象。队列中 允许插入的一端叫队尾…

快速理解微服务中Sentinel怎么实现限流

Sentinel是通过动态管理限流规则&#xff0c;根据定义的规则对请求进行限流控制。 一.实现步骤 1.定义资源&#xff1a;在Sentinel中&#xff0c;资源可以是URL、方法等&#xff0c;用于标识需要进行限流的请求&#xff1b;(在Sentinel中&#xff0c;需要我们去告诉Sentinel哪些…

matlab根据excel表头筛选表格数据

有如下表格需要筛选&#xff1a; 如果要筛选style中的A&#xff0c;color中的F2&#xff0c;num中的3。 代码如下&#xff1a; clear;clc; file_Pathstrcat(F:\csdn\,test1.xlsx); %表格路径、文件名 E1readtable(file_Path,Sheet,1); %读取表格中的字母和数字,1代表第一个…

学习日志016--python实现双向循环列表与链栈

python中一些复合数据结构通过类的封装来实现的。双向循环链表与链栈也在其中。 双向循环链表 双向循环链表是一种特殊类型的链表&#xff0c;它结合了双向链表和循环链表的特点。在双向循环链表中&#xff0c;每个节点不仅包含数据&#xff0c;还持有指向前一个和后一个节点的…