数据结构链表(C语言实现)

news2024/9/23 7:31:04

绪论

        机遇对于有准备的头脑有特别的亲和力。本章将讲写到链表其中主要将写到单链表和带头双向循环链表的如何实现。

 话不多说安全带系好,发车啦(建议电脑观看)


附:红色,部分为重点部分;蓝颜色为需要记忆的部分(不是死记硬背哈,多敲);黑色加粗或者其余颜色为次重点;黑色为描述需要


目录

链表

1.单链表   

1.1.单链表的结构

1.2放入数据到单链表中

1.3 删除单链表中的数据

1.4 摧毁单链表

1.5单链表的打印

2.带头双向循环列表

2.1带头双向链表的结构

2.2双向链表的初始化

2.3双向链表中放入数据

2.4双向链表中的删除数据

2.5双向链表的摧毁

2.6打印双向链表中的各个数据

3.如何快速的实现一个链表


链表

知识点:

链表是一个在逻辑上连续的线性结构,但在物理结构上他的地址并不是连续的,是通过指针的形式链接而成的一个非顺序的存储结构。

细节:

链表的属性:

单向/双向、循环/不循环、带头/不带头

对此就能有多种排列组合,但我们主要学的是:单链表(单向、不带头、不循环)、带头双向循环链表(因为这两个链表最具代表性且常用,当我们学会这两种链表后其他的链表基本也不在话下)

对于上面的属性其实也很好理解下面通过一张图片具体展示:

其中节点表示的就是每个数据的每个小空间图中的矩形


1.单链表   

单链表是我们在链表中最简单的一种链表形式,因为过于简单所以其一般并不会用于存储数据,但却经常出现于习题以及笔试面试题中,并且还会作为一些数据结构的子结构(如:图的领接表、哈希桶中)。

单链表的基本框架:

1. 单链表的结构

        1.一个数剧date(存储当前位置的数据)

        2.一个指针next(指向下一个数据)

2. 所要实现的功能

        1.将数据放入结构中

        2.查找单链表中的某个的数据

        3.将数据从结构中删除(删除节点,前提是要先找到该节点所在的位置)

        4.将数据展现打印出来

        5.摧毁单链表

1.1.单链表的结构

单链表的是由数据、指针组成:

typedef struct SListNode
{
	SLDateType data;//SLDateType typedef定义类型和顺序表一样都是为了更加方便去改变结构中的存的类型
	struct SListNode* next;//定义一个结构体类型的指针,因为到时候next指向一个结构体类型
}SListNode;//用typedef将struct SListNode改变成SListNode这样更方便于我们后面使用(可以不用加上struct,直接用代替的就好)

//typedef int SLDateType; int重命名为SLDateType

1.2放入数据到单链表中

在插入数据之前需要先开辟好空间(也就是创建节点):

SListNode* BuySListNode(SLDateType x) {//接收传进来的数据
	SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));//用malloc开辟一个结构体大小的空间给新节点
	if (newnode == NULL)//熟系的判断是否申请成功
	{
		perror("malloc failed");
		return NULL;
	}
	newnode->data = x;//在申请的结构体空间中放入数据
	newnode->next = NULL;//将其下一个位置暂时指向NULL
	return newnode;//返回这个节点,返回类型是个指针类型也就是newnode的地址
}

申请好空间后就可以进行插入数据(并且链接了)

  1. 尾插:从尾部插入,分析尾部的特征是其下一个指向的是NULL对此就能找到最后一个数据的位置将其指向的NULL改成新节点的位置即可
    void SListPushBack(SListNode** pphead, SLDateType x) {//注意此处用二级指针是因为可能要修改结构体变量(或理解成传进来的是一级指针类型为了修改他就需要用二级指针)
    	assert(pphead);//判空,注意此处的判空是判的pphead 而不是 STL(*pphead) 
    	SListNode* newnode = BuySListNode(x);//先创建一个新节点,调用创建节点的函数
    	SListNode* tail = *pphead;//一个指针指向链表开始(可能为NULL)
    	
    	if (*pphead == NULL)//此处就是一个类似初始化的步骤
    	{
    		*pphead = newnode;//此处修改了了结构体变量,即修改了STL的指向指向新创建的链表的开始
    	}
    
    	else {//当不是最开始时就正常的进行尾插	
    		while (tail->next) {//找尾部的NULL
    			tail = tail->next;//没找到就改变tail 让其变成next 继续往后找
    		}
    		tail->next = newnode;//找到后把最后一个位置的数据的next 变成 新节点即可
    	}
    }
    动图展示具体步骤:
  2. 头插,比较简单直接看注释就能理解
    //头插就会简单很多
    void SListPushFront(SListNode** pphead, SLDateType x){//同样用到二级指针,因为需要改变链表结构体变量
    	assert(pphead);
    	SListNode* newnode = BuySListNode(x);//创建节点
    
    	newnode->next = *pphead;//将新开辟的空间的地址指向链表原本第一个元素的 地址
    	*pphead = newnode;//改变链表的起始地址
    }

从指定位置插入数据:

单独拎出来是因为其比较特殊,它可以分为在pos位置前插入数据和在pos位置后插入数据两种,并且我们在使用前需要先找到指定的位置后才能进行数据的插入(所以还需要先写一个查找数据的函数接口)。查找单链表中的某个数据(思路类似于尾插):

SListNode* SListFind(SListNode* phead, SLDateType x) {//指针接收结构体
	SListNode* find = phead;
	while (find)//一个循环来找想要的数据
	{
		if (find->data == x)//当找到时
		{
			return find;//返回此处的地址
		}
		find = find->next;//不断往后找和找尾方法一样
	}
	printf("找不到\n");
	return NULL;//若找不到返回NULL
}

有了之后我们就可以利用这个查找函数还进行对该数据的前/后插入数据了:

  1. 先向pos位置前插入(一般不考虑这种,因为比较麻烦且效率低,多在后面插入数据):
    void SListInsertFront(SListNode** pphead,SListNode* pos, SLDateType x) {//当是在pos位置前面插入时就要考虑头插了所以可能会改变链表的开始就需要用到二级指针
    	assert(pphead);
    	assert(pos); 
    	if (pos == *pphead)//如果pos位置在第一个那就等于在第一个位置前面插入数据也就是头插
    	{
    		SListPushFront(pphead, x);//此处用pphead因为pphead已经是一个二级指针了
    	}
    	else
    	{
    		SListNode* tail = *pphead;
    		while (tail->next != pos)//找到pos位置的前一个位置
    		{
    			tail = tail->next;
    		}
    		SListNode* newnode = BuySListNode(x);//申请一个节点
    		newnode->next = tail->next;//先把newnode的指针改变,这样避免当tail改变后找不到tail->next的情况
    		tail->next = newnode;//再将tail->next 指向改变成 newnode
    	}
    }
  2. 先pos位置之后插入数据:
    void SListInsertAfter(SListNode* pos, SLDateType x) {//此处把链表的地址传给了pos
    	assert(pos);
    	SListNode* newnode = BuySListNode(x);//创建新节点
    	newnode->next = pos->next;//先改变newnode的节点
    	pos->next = newnode;//再把pos->next改成newnode
    }
  3. 附:当不给头怎么才能在pos位置前面的插入数据,有一种思路现在其pos位置后面插入一个数据然后再将pos位置的值和后面插入的值进行交换就相当于在pos位置前插入数据了。

1.3 删除单链表中的数据

  1. 尾删:找到倒数第二个数据的位置后再把最后一个位置的空间进行释放即可,然后把倒数第二个位置的next指向NULL。
    void SListPopBack(SListNode** pphead) {//可能会修改外部数据所以用二级指针
    	assert(pphead);//判空
    	assert(*pphead);//判空,防止第一个位置都没有
    	if ((*pphead)->next == NULL)//查看第一个数据位置是不是最后一个数据如果是的话
    	{
    		free(*pphead);//直接释放第一个位置的数据即可
    		*pphead = NULL;//把没用的指针置为NULL
    	}
    	else
    	{
    		SListNode* tail = *pphead;//用tail代替*pphead,使*pphead不被改变
    
    		while (tail->next->next) {//往后看两位,当为空时,就表示到了倒数第二的位置
    			tail = tail->next;//一步一步走
    		}
    
    		free(tail->next);//找到后,tail表示的是倒数第二,而tail->next就是尾,free释放尾即可
    		tail->next = NULL;//tail->next改变成NULL,因为此时tail变成了最后一个数据
    	}
    }
  2. 头删:分析首先我们肯定是需要一个二级指针接收来改变链表的开始,其次就是要用一个临时指针指向头的位置先把头改成下一个位置,然后把临时变量指向的原本的头的空间释放

    void SListPopFront(SListNode** pphead) {
    	assert(pphead);//防止pphead为空
    	assert(*pphead);//查看是否有第一个数据
    
    	SListNode* frist = *pphead;//记录第一个数据的位置
    	*pphead = frist->next;//改变链表的头,改成下一个位置(可能为空当只有一个数据的时候)
    	free(frist);//(释放第一个数据的位置)
    	frist = NULL;//没有的指针置为空
    }
    

在pos位置/pos位置后删除数据

在前面已经提过在pos位置前面插入基本不会用到所以此处就不再写了

  1. 在pos位置处删除,需要先找到pos位置前面的数据然后后才能改变链接关系
    void SListErase(SListNode** pphead,SListNode* pos) {
    	assert(pos);//不用再对*pphead进行检查了因为pos已经间接的检查了*pphead 检查的是是否链表中是否有数据假如没有的话pos也会报错
    	assert(pphead);//判空
    	SListNode* tail = *pphead;
    	if (*pphead == pos)//若是第一个元素
    	{
    		SListPopFront(pphead);//那就直接头删
    	}
    	else
    	{
    		while (tail->next != pos)//找到pos位置的前一个位置
    		{
    			tail = tail->next;
    		}
    		tail->next = pos->next;//将前面位置的next改变成pos位置的next,这样就把pos位置给断开了
    		free(pos);//把pos位置处的空间给释放
    	}
    }
  2. 把pos位置后面的数据删除,方法很简单同样就是改变链接关系即可即吧pos->next = pos->next->next(即先找到pos位置后面的数据,然后把pos的next改成后面数据的next即可思想和上面的是一样的)
    void SListEraseAfter(SListNode* pos) {
    	assert(pos);
    	if (pos->next == NULL)//注意查看pos位置后面是否还有数据
    	{
    		return;//若没有则直接返回了
    	}
    	else {
    		SListNode* del = pos->next;//记录pos后面的位置
    		pos->next = pos->next->next;//改变链接关系让pos的后面位置指向pos位置后的后面位置
    		free(del);//释放del
    		del = NULL;
    	}
    }
  3. 若没给头怎么删除pos位置处的数据呢?可以先存一份pos位置后面的数据再将后面的数据给赋给pos位置再把pos位置后面的数据给删除即可。

1.4 摧毁单链表

从前往后的依次free(单链表不能从后往前是因为找不到是前面的数据的),并且注意还需要一个指针来在后面,才能保证释放前面数据后能找到后面的数据。

void SListDestroy(SListNode* plist) {
	assert(plist);

	SListNode* prev, * tail;//双指针
	prev = plist;//指向头
	tail = plist->next;//指向第二个数据的位置
	while (prev)//当prve==NULL就不用进循环了也表示释放完了
	{	
		free(prev);//释放prev出的空间
		prev = tail;//将prve指向tail
		if(tail != NULL)//判断tail是不是NULL
			tail = tail->next;//若是NULL就不能再往后了
	}
	plist = NULL;//将没用的指针赋成空指针
}

1.5单链表的打印

直接遍历整个数组然后找到其数据进行打印

void SListPrint(SListNode* phead)
{
	SListNode* tail = phead;//用一个指针来指向开始

	while (tail != NULL)//只要tail不到NULL都要进群
	{
		printf("%d->", tail->data);//进来打印tail的date
		tail = tail->next;//往后走
	}
	printf("NULL\n");//打印一下最后的NULL
}

附:为什么要用二级指针?

因为如果不用二级指针的话就无法改变传进来的参数(只是传调用无法改变参数),所以只有用传址调用能改变形参,而传递进来的是一个一级指针类型,所以我们就需要二级指针类型来接收他的地址(接收指针类型的指针三步法)。


2.带头双向循环列表

知识点:

本质还是一个链表所以还是一样的通过指针的方法链接,但是需要我们去改变一下结构,增加了一个prev指针指向前面的数据。结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。

细节:

带头双向循环链表的基本框架:

  1. 带头双向循环链表的结构
    1. 一个变量存数据
    2. 一个指针指向前面的数据
    3. 一个指针指向后面的数据
  2. 带头双向循环链表所要实现的功能
    1. 初始化结构
    2. 将数据放进结构中
    3. 将数据从结构中删除
    4. 在指定位置处插入/删除数据

2.1带头双向链表的结构

它的结构是由存数据的变量、两个分别指向前后的指针组成的

typedef int LTDataType;

typedef struct ListNode
{
	LTDataType data;//数据
	struct ListNode* next;//指向后面的数据
	struct ListNode* prev;//指向前面的数据
	//struct ListNode结构体类型,这里要加上struct因为类型重命名是在后面的
}ListNode;//typedef类型重命名

2.2双向链表的初始化

在那之前还需要去写一个申请空间的接口

ListNode* BuyMemory(LTDataType  x)//申请空间的函数
{
	ListNode* node = (ListNode*)malloc(sizeof(ListNode));//malloc申请一个大小为一个结构体(一个节点的)空间
	if (node== NULL)//若申请失败
	{
		perror("malloc::BuyMemory");//报错
		//return NULL;
		exit(-1);//直接退出程序
	}
	node->data = x;//将申请中的空间数据置为给定的x
	node->next = NULL;//将next先置为NULL
	node->prev = NULL;//将prev先置为NULL

	return node;//返回借号的空间
}

因为是带头的链表所以需要去先申请一个空间给头

ListNode* ListCreate()//给头申请空间并且返回头指针的地址
{
	ListNode* head = BuyMemory(-1);//申请空间并且将其数据置为-1
		head->next = head;//将next指向head
		head->prev = head;//prev也是指向头的
	return head;//返回头申请好的空间
}

此处因为有了头,那就不再需要先去创建一个链表的结构(ListNode *STL = NULL )来确定链表的头了(不用传STL/&STL、而是直接传head即可找到链表并且修改链表(即使是改变链表的开始也不用再使用二级指针了))。

下面的头插、头删中的头并不是头而是表示插入/删除第一个节点


2.3双向链表中放入数据

  1. 头插思路:先记录通过head找到的原本的头再创建一个新节点改变链接关系即可完成
    
    void ListPushFront(ListNode* pHead, LTDataType x)
    {
    	assert(pHead);//判空
    	ListNode* newnode = BuyMemory(x);//创建新节点
    	ListNode* frist = pHead->next;//记录原本的链表的第一个数据
    	//改变链接关系
    	// phead <-> first
    	// phead <-> newnode <-> first
    	//先改变newnode与第一个数据间的关系,
    	// newnode <-> firest
    	newnode->next = frist; 
    	frist->prev = newnode;
    	//再改变新节点个head的链接关系
    	// head <-> newnode
    	newnode->prev = pHead;
    	pHead->next = newnode;
    	//改完后 :phead <-> newnode <-> frist
    }

  2. 尾插方法和头插类似,先记录尾部数据,然后新建一个节点,改变head、尾部数据、新节点三者的链接关系
    void ListPushBack(ListNode* pHead, LTDataType x)
    {
    	assert(pHead);
    
    	ListNode* newnode = BuyMemory(x);//创建新节点
    	ListNode* tail = pHead->prev;//记录原本的尾部数据结构
    	//原本的链接顺序:tail <-> head
    	//改变后的链接顺序:tail <-> newnode <-> head
    	
    	//先改变 tail <-> newnode
    	tail->next = newnode;
    	newnode->prev = tail;
    	//再把newnode和head链接起来:newnode<->head
    	newnode->next = pHead;	
    	pHead->prev = newnode;
    }
  3. 在在pos位置插入数据,因为这是一个双向且循环的链表,所以我们只需要写一个插入接口即可实现所有想要的结果(此处实现在pos位置前面插入数据,其实现方法和上面的都几乎一样)。

    void ListInsert(ListNode* pos, LTDataType x)
    {
    	assert(pos);//判空
    	ListNode* prev = pos->prev;//记录前面的数据
    	ListNode* newnode = BuyMemory(x);//创建新节点
    
    	//改变链接关系
    	// prev pos
    	// prev newnode pos
    	
    	// prev newnode 
    	prev->next = newnode;
    	newnode->prev = prev;
    	// newnode pos
    	newnode->next = pos;
    	pos->prev = newnode;
    
    // prev newnode pos
    }

    同样的为了获取pos的结构的空间地址,我们需要写一个查找的接口

    ListNode* ListFind(ListNode* pHead, LTDataType x)
    {
    
    	assert(pHead);
    
    	ListNode* tail = pHead;//tail指向链表
    	while (tail)//通过tail指针来找
    	{
    		if (tail->data == x)//找结构中的值是不是等于x
    		{
    			return tail;//找到后返回节点的地址
    		}
    		tail = tail->next;//往后走
    	}
    	printf("不存在,找不到\n"); //若到了此处就表示链表中的数据没有x
    	return NULL;//此时返回NULL表示不存在该数据的节点
    }

2.4双向链表中的删除数据

在我们删除数据的前提是得链表中有数据所以就需要一个判断是否有数据的接口:

bool If_DTLEmpty(ListNode* pHead)//返回bool值,即返回真或假
{
	if (pHead->next == pHead)//就判断phead的->是不是自己即可,因为如果不是则表示是有数据的,反之因为是循环链表没数据就会指向自己了
	{
		return true;//若指向自己则是真 空的
	}
	else
	{
		return false;//反之则非空
	}
}
  1. 头删,记录第一、二个节点,改变链接关系即可
    void ListPopFront(ListNode* pHead)
    {
    	assert(pHead);
    	assert(!If_DTLEmpty(pHead));//判断是否为空
    	ListNode* front = pHead->next;//记录第一个节点
    
    
    	ListNode* frist = pHead->next->next;//记录第二个节点
    	//改变链接关系
    	// head <-> front <-> frist  ==  head <-> first
    	pHead->next = frist;
    	frist->prev = pHead;
    
    	
    	free(front);//改变链接关系后把第一个节点释放
    	front = NULL;
    }
  2. 尾删,记录倒数第一、二个节点,改变链接关系即可
    void ListPopBack(ListNode* pHead)
    {
    	assert(pHead);
    	assert(!If_DTLEmpty(pHead));//判断是否为空
    	ListNode* new_tail = pHead->prev->prev;//记录倒数第二个节点
    	ListNode* old_tail = pHead->prev;//记录倒数第一个节点
    
    	//改变链接关系
    	new_tail->next = pHead;
    	pHead->prev = new_tail;
    
    	free(old_tail);//释放倒数第一个节点
    	old_tail = NULL;
    }
  3. 删除指定pos位置处的数据,记录pos位置的next和prev节点然后改变链接关系即可
    void ListErase(ListNode* pos)
    {
    	assert(pos);//判空
    	ListNode* prev = pos->prev;//记录pos前面节点
    	ListNode* tail = pos->next;//记录pos后面的节点
    	//改变链接关系
    	prev->next = tail;
    	
    	tail->prev = prev;
    	//释放pos
    	free(pos);
    }

2.5双向链表的摧毁

其摧毁的方法和单链表很像,但要注意他是一个循环的链表,区别在于多了个头所以就可以以这个头为结束,从第一个节点开始,当遇到头时就表示走完一遍链表了就跳出循环,最后再把head释放一下即可

void ListDestory(ListNode* pHead)
{
	assert(pHead);
	ListNode* tail = pHead->prev;//从第一个节点开始
	while (tail != pHead)//当没遇到头前就不断释放再往前走
	{
		ListNode* destroy = tail;//记录要释放的节点地址
		tail = tail->next;//tail往后走
		free(destroy);//释放空间
	}
	free(pHead);//最后当把链表中的数据释放完后,再释放头
}

2.6打印双向链表中的各个数据

从第一个节点开始遍历链表,当循环回来到head时就退出

void ListPrint(ListNode* pHead)
{
	assert(pHead);
	ListNode* cur = pHead->next;//从第一个节点开始
	printf("《=》head《=》");

	while (cur != pHead)//当不是pHead就进去
	{
		printf("%d《=》", cur->data);//打印
		cur = cur->next;//往后走
	}
	printf("\n");
}

3.如何快速的实现一个链表

其实如果要快速的实现一个链表的话,用到的就是双向循环链表,因为其是双向的这样就可以直接通过在pos位置插入/删除,这样我们就能剩下写头插、删、尾插、删的时间直接通过插入/删除接口间接实现

ListNode* BuyMemory(LTDataType  x)//申请空间的函数
{
	ListNode* node = (ListNode*)malloc(sizeof(ListNode));//malloc申请一个大小为一个结构体(一个节点的)空间
	if (node== NULL)//若申请失败
	{
		perror("malloc::BuyMemory");//报错
		//return NULL;
		exit(-1);//直接退出程序
	}
	node->data = x;//将申请中的空间数据置为给定的x
	node->next = NULL;//将next先置为NULL
	node->prev = NULL;//将prev先置为NULL

	return node;//返回借号的空间
}

ListNode* ListCreate()//给头申请空间并且返回头指针的地址
{
	ListNode* head = BuyMemory(-1);//申请空间并且将其数据置为-1
		head->next = head;//将next指向head
		head->prev = head;//prev也是指向头的
	return head;//返回头申请好的空间
}

void ListPushBack(ListNode* pHead, LTDataType x)
{
	assert(pHead);
	ListInsert(pHead,x);
}


void ListPrint(ListNode* pHead)
{
	assert(pHead);
	ListNode* cur = pHead->next;//从第一个节点开始
	printf("《=》head《=》");

	while (cur != pHead)//当不是pHead就进去
	{
		printf("%d《=》", cur->data);//打印
		cur = cur->next;//往后走
	}
	printf("\n");
}

bool If_DTLEmpty(ListNode* pHead)//返回bool值,即返回真或假
{
	if (pHead->next == pHead)//就判断phead的->是不是自己即可,因为如果不是则表示是有数据的,反之因为是循环链表没数据就会指向自己了
	{
		return true;//若指向自己则是真 空的
	}
	else
	{
		return false;//反之则非空
	}

}


void ListPopBack(ListNode* pHead)
{
	assert(pHead);
	assert(!If_DTLEmpty(pHead));//判断链表是否为空
	ListErase(pHead->prev);
}


void ListPushFront(ListNode* pHead, LTDataType x)
{
	assert(pHead);//判空
	ListInsert(pHead->next,x);
}



void ListPopFront(ListNode* pHead)
{
	assert(pHead);
	assert(!If_DTLEmpty(pHead));//判断是否为空
	ListErase(pHead->next);
}

ListNode* ListFind(ListNode* pHead, LTDataType x)
{

	assert(pHead);

	ListNode* tail = pHead;//tail指向链表
	while (tail)//通过tail指针来找
	{
		if (tail->data == x)//找结构中的值是不是等于x
		{
			return tail;//找到后返回节点的地址
		}
		tail = tail->next;//往后走
	}
	printf("不存在,找不到\n"); //若到了此处就表示链表中的数据没有x
	return NULL;//此时返回NULL表示不存在该数据的节点
}


void ListInsert(ListNode* pos, LTDataType x)
{
	assert(pos);//判空
	ListNode* prev = pos->prev;//记录前面的数据
	ListNode* newnode = BuyMemory(x);//创建新节点

	//改变链接关系
	// prev pos
	// prev newnode pos
	
	// prev newnode 
	prev->next = newnode;
	newnode->prev = prev;
	// newnode pos
	newnode->next = pos;
	pos->prev = newnode;

// prev newnode pos
}


void ListErase(ListNode* pos)
{
	assert(pos);//判空
	ListNode* prev = pos->prev;//记录pos前面节点
	ListNode* tail = pos->next;//记录pos后面的节点
	//改变链接关系
	prev->next = tail;
	
	tail->prev = prev;
	//释放pos
	free(pos);
}

void ListDestory(ListNode* pHead)
{
	assert(pHead);
	ListNode* tail = pHead->prev;//从第一个节点开始
	while (tail != pHead)//当没遇到头前就不断释放再往前走
	{
		ListNode* destroy = tail;//记录要释放的节点地址
		tail = tail->next;//tail往后走
		free(destroy);//释放空间
	}
	free(pHead);//最后当把链表中的数据释放完后,再释放头
}



如果有任何问题欢迎讨论哈!

如果觉得这篇文章对你有所帮助的话点点赞吧!

持续更新大量数据结构细致内容,早关注不迷路。

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

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

相关文章

【微服务】一文了解Nginx网关搭建教程

一文了解Nginx网关搭建教程 Nginx网关搭建nginx配置Nginx网关搭建 那么什么是Nginx呢? nginx是一个高性能HTTP服务器,反向代理服务器,邮件代理服务器,TCP/UDP反向代理服务器。 单个系统主要用于处理客户端请求,一个系统处理客户端的请求量是有限的,当客户端的并发量超…

CROSSROADS: 1实战演练

文章目录 CROSSROADS: 1实战演练一、前期准备1、相关信息 二、信息收集1、端口扫描2、访问网站3、dirsearch扫描目录4、查看隐写5、枚举用户6、暴力破解7、访问共享文件夹8、查看第一个flag9、写入shell并连接 二、后渗透1、查看权限和SUID文件2、运行程序3、切换root3、查看第…

2019 - 2023,再见了我的大学四年

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;也会涉及到服务端&#xff08;Node.js 等&#xff09; &#x1f4c3;个人状态&#xff1a; 2023届本科毕业生&#xff0c;已拿多个前端 offer&#x…

【Linux环境基础开发工具】编译器-gcc/g++

写在前面&#xff1a; 上一篇博客&#xff0c; 我们学习了vim编辑工具&#xff0c;学会了怎么写代码&#xff0c; 这篇文章&#xff0c;我将分享代码该怎么编译的问题。 目录 写在前面&#xff1a; 1. gcc和g介绍 2. gcc是如何编译程序的 1. 预处理 2. 编译 3. 汇编 …

CPM-Bee大模型微调

CPM-Bee大模型微调 CPM-Bee简介&#xff1a;环境配置&#xff1a;应用场景&#xff1a;模型训练参数训练命令&#xff1a;推理&#xff1a;评估&#xff1a;结论&#xff1a; CPM-Bee 简介&#xff1a; CPM-Bee是一个完全开源、允许商用的百亿参数中英文基座模型&#xff0c;也…

关于数据库运维系统的一些思考

这是学习笔记的第 2461篇文章 前段时间整理了一下数据库运维系统的一些内容&#xff0c;比自己预期的要难一些。我来简单回顾下一些参考点。 一、立足当下&#xff0c;混沌之中梳理问题 通常我们可以会问为什么&#xff0c;即为什么要做数据库运维系统&#xff0c;但是我们先放…

决策分析——层次分析法

工程测量与经济决策方案 决策分析——层次分析法 一、描述 层次分析法的基本原理&#xff1a;根据问题的性质和要达到的总目标&#xff0c;将问题分解为不同的组成因素&#xff0c;并按照因素间的相互关联影响以及隶属关系将因素按不同层次聚集组合&#xff0c;形成一个多层次…

NUCLEO-F411RE RT-Thread 体验 (8) - GCC环境 TIM定时器的驱动移植以及基本使用

NUCLEO-F411RE RT-Thread 体验 (8) - GCC环境 TIM定时器驱动移植与基本使用 驱动移植 定时器驱动文件位于drv_hwtimer.c中&#xff0c;对应components层的文件位于rt-thread/components/drivers/hwtimer/hwtimer.c中。 修改Makefile&#xff0c;将其编译进去。 在rtconfig.h…

Xception算法解析-鸟类识别实战-Paddle实战

文章目录 项目背景一、理论基础1.前言2.设计理念2.1 多尺寸卷积核2.2 点卷积2.3 卷积核替换2.4 Bottleneck2.5 深度可分离卷积(Depthwise Separable Conv) 3.网络结构4.评估分析 二、数据预处理三、数据读取四、导入模型五、模型训练六、结果可视化七、个体预测结果展示总结 项…

BLE蓝牙模块应用|蓝牙MAC地址二维码扫描打印解决方案

在智能穿戴、智能安防领域&#xff0c;用户需要通过蓝牙MAC地址来完成产品与APP的绑定。为简化产品的使用和管理&#xff0c;厂家会采用蓝牙MAC地址二维码扫描打印解决方案&#xff0c;将蓝牙MAC地址打印成二维码并贴在产品的包装盒或者产品外壳上。本篇带大家了解一下蓝牙MAC地…

基于Java德云社票务系统设计实现(源码+lw+部署文档+讲解等)

博主介绍&#xff1a; ✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战 ✌ &#x1f345; 文末获取源码联系 &#x1f345; &#x1f447;&#x1f3fb; 精…

干货 | 电力数据流通使用模式及安全指南

以下内容整理自清华大学《数智安全与标准化》课程大作业期末报告同学的汇报内容。 一、电力数据及流通简介 电力数据的流通整体环节也是它生命的全周期。 电力用户大体可以分为四类&#xff0c;分别是个人&#xff0c;工农业企业&#xff0c;商业建筑以及城市基建&#xff0c;这…

火山引擎 Dataleap 数据质量解决方案和最佳实践(一):数据质量挑战

更多技术交流、求职机会&#xff0c;欢迎关注字节跳动数据平台微信公众号&#xff0c;回复【1】进入官方交流群 什么是数据质量 广义上来说&#xff0c;数据质量的定义是数据满足一组固有特性&#xff08;质量维度&#xff09;要求的程度。业界通常有 6 个维度&#xff1a; 完…

语法降级与Polyfill:消灭低版本浏览器兼容问题

提到前端编译工具链方面&#xff0c;可能大家最新想到的是诸如babel/preset-env、core-js、regenerator-runtime等工具。不过&#xff0c;我们今天要讲的是官方的 Vite 插件vitejs/plugin-legacy&#xff0c;以及如何将这些底层的工具链接入到 Vite 中&#xff0c;并实现开箱即…

【强化学习】——Q-learning算法为例入门Pytorch强化学习

&#x1f935;‍♂️ 个人主页&#xff1a;Lingxw_w的个人主页 ✍&#x1f3fb;作者简介&#xff1a;计算机研究生在读&#xff0c;研究方向复杂网络和数据挖掘&#xff0c;阿里云专家博主&#xff0c;华为云云享专家&#xff0c;CSDN专家博主、人工智能领域优质创作者&#xf…

神经网络:参数更新

在计算机视觉中&#xff0c;参数更新是指通过使用梯度信息来调整神经网络模型中的参数&#xff0c;从而逐步优化模型的性能。参数更新的作用、原理和意义如下&#xff1a; 1. 作用&#xff1a; 改进模型性能&#xff1a;参数更新可以使模型更好地适应训练数据&#xff0c;提高…

python学习——pandas统计分析基础

目录 pandas统计分析基础1. Series数据2.文件读取csv文件Excel文件 3.DataFrame连接数据库读取数据库存入数据库DataFrame的属性访问DataFrame中的数据【实例1】info详细信息和describe描述统计分析【实例2】 排序【实例3】 布尔索引&#xff0c;条件索引【案例】修改数据 3.描…

LIBSVM与LIBLINEAR支持向量机库对模式识别与回归的可视化代码实践

支持向量机(SVM)是一种流行的分类技术。虽然提出时间到现在有70来年了&#xff0c;但在90年代获得了很好的发展和扩展&#xff0c;在人像识别、文本分类、手写字符识别、生物信息学等模式识别问题中有得到应用。然而&#xff0c;对于不熟悉SVM的初学者来说&#xff0c;往往会因…

ThreadPoolExecutor解读

目录 线程池状态 构造方法 newFixedThreadPool newCachedThreadPool newSingleThreadExecutor 提交任务 关闭线程池 其它方法 线程池状态 ThreadPoolExecutor 使用 int 的高 3 位来表示线程池状态&#xff0c;低 29 位表示线程数量 状态名 高 3 位 接收新任务 处理…

JavaScript ES10新特性

文章目录 导文Array.prototype.flat()和Array.prototype.flatMap()Object.fromEntries()String.prototype.trimStart()和String.prototype.trimEnd()格式化数字动态导入可选的catch绑定BigIntglobalThis 导文 JavaScript ES10&#xff0c;也被称为ES2019&#xff0c;引入了一些…