【数据结构】单链表专题详细分析

news2025/1/11 18:49:36

与其临渊羡鱼,不如退而结网。💓💓💓

目录

•✨说在前面

🍋知识点一:什么是链表?

 • 🌰1.链表的概念 

 • 🌰2.链表的结构

 • 🌰3.链表的分类

🍋知识点二:单链表

 • 🌰1.顺序表的劣势

 • 🌰2.单链表动态申请节点

 • 🌰3.链表元素的打印

 • 🌰4.单链表头部插入元素

 • 🌰5.单链表尾部插入元素

 • 🌰6.指定位置之前插入数据

 • 🌰7.指定位置之后插入数据

 • 🌰8.单链表头部删除元素

 • 🌰9.单链表尾部删除元素

 • 🌰10.删除指定位置的节点

 • 🌰11.删除指定位置之后的节点

 • 🌰12.单链表的查找

 • 🌰13.单链表的销毁

🍋知识点三:单链表基本操作

• ✨SumUp结语


•✨说在前面

亲爱的读者们大家好!💖💖💖,我们又见面了,在之前的学习中我们了解了什么是数据结构以及顺序表的基本操作,而且还用顺序表做了通讯录的小项目。然而,顺序表并不是完美的,它的操作容易浪费空间,甚至降低程序的运行效率。我们今天要学习的链表,在这些地方就优于顺序表。

  

如果你没有准备好的话,或者说你还没有办法独立完成顺序表功能的代码,希望你先回去看看顺序表部分的内容,确认自己没有问题之后再来看这篇文章。

   

👇👇👇
💘💘💘知识连线时刻(直接点击即可)

  🎉🎉🎉复习回顾🎉🎉🎉

        【数据结构】顺序表专题详解(带图解析)

        【数据结构】基于顺序表实现通讯录

    

  博主主页传送门:愿天垂怜的博客

🍋知识点一:什么是链表?

 • 🌰1.链表的概念 

定义通讯录顺序表链表是一种线性数据结构,由一系列的节点组成,每个节点都包含数据和指向下一个节点的指针。

链表的元素在内存中不必连续排列,而是通过指针相互连接。

 链表的结构和火车箱相似,淡季时车次的车厢会相应减少,旺季时车次的车厢会额外增加几节,只需要将火车的某节车厢去掉/加上,不会影响其他车厢,每节车厢都是独立存在的。

设每节车厢的车门都是锁上的状态,需要不同的钥匙才能解锁,每次只能携带一把钥匙的情况下如何从车头走到车尾?

最简单的做法:每节车厢里都放一把下一节车厢的钥匙。

在链表里,每节"车厢"是什么样的呢?

 与顺序表不同的是,链表里的每节"车厢"都是独立申请下来的空间,我们称之为"结点/节点"。节点的组成主要有两个部分:当前节点要保存的数据和保存下一个节点的地址(指针变量)。

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

📌为什么还需要指针变量来保存下一个节点的位置?

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

 • 🌰2.链表的结构

链表的基本结构由节点组成,每个节点包含数据和执行下一个节点的指针。

结合之前结构体的知识,我们可以写出节点数据为整型的链表:

typedef int SLTDataType;

typedef struct SListNode
{
	SLTDataType data;//数据域
	struct SListNode* next;//指针域
}SLTNode;

 • 🌰3.链表的分类

链表可以分为单向链表、双向链表和循环链表。

🎉单向链表:每个节点只有一个指针指向下一个节点。

🎉双向链表:每个节点有两个指针,分别指向前一个节点和后一个节点。

 🎉循环链表:尾节点指向头结点。

链表的结构非常多样,以下情况组合起来就有8种(2 x 2 x 2) 链表结构:

链表说明: 

🍋知识点二:单链表

 • 🌰1.顺序表的劣势

🎉中间/头部的插入删除,时间复杂度为O(N)

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

🎉增容一般是2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们再进行插入了5个数据,后面就没有数据插入了,那么就浪费了95个数据空间。

总结:

中间/头部插入删除效率低下、增容降低运行效率、增容造成空间浪费。

然而我们接下来要学习的链表就完美地解决了顺序表的问题。

 • 🌰2.单链表动态申请节点

单链表申请节点的过程其实就是初始化的过程,也就是说单链表不需要单独初始化,需要用单链表的时候我们直接申请节点就可以了,当然,不论是头插、尾插还是指定位置插入都离不来申请节点的操作,所以我们将它写成一个函数:

SLTNode* SLTBuyNode(SLTDataType x)
{
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	if (newnode == NULL)
	{
		perror("malloc");
		exit(1);
	}
	newnode->data = x;
	newnode->next = NULL;

	return newnode;
}

申请一个结点newnode,首先if判空,不为空就将x存入数据域data,再让newnode的next指针置为NULL,因为没有下一个节点,所以必须置空,否则为野指针

 • 🌰3.链表元素的打印

注意:链表和顺序表是有所不同的,顺序表传过来的指针是肯定不会为空的,而链表传过来的指针是可能为空的比如说当链表中没有元素时,头指针所指向的就是NULL,如果在第一行写上断言就会有问题,所以不需要assert断言。

void SLTPrint(SLTNode* phead)
{
	SLTNode* pcur = phead;
	while (pcur)
	{
		printf("%d->", pcur->data);
		pcur = pcur->next;
	}
	printf("NULL\n");
}

例如下列链表:

int main()
{
    //创建链表节点
	SLTNode* node1 = SLTBuyNode(1);
	SLTNode* node2 = SLTBuyNode(2);
	SLTNode* node3 = SLTBuyNode(3);
	SLTNode* node4 = SLTBuyNode(4);
    
    //让next指针指向下一个节点,最后一个为NULL
	node1->next = node2;
	node2->next = node3;
	node3->next = node4;
	node4 = NULL;

	SLTPrint(node1);

	return 0;
}

 利用SLTPrint函数得到了结果如下:

1->2->3->4->NULL

 • 🌰4.单链表头部插入元素

向单链表的头部插入元素,我们只需要创建一个新节点,让这个新节点指向我们原来的第一个节点,再将第一个节点的指针*pphead指向newnode即可。

void SLTPushFront(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	SLTNode* newnode = SLTBuyNode(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

注意,对于单链表来说,单链表的头部位置插入/删除、尾部位置插入/删除、指定位置插入/删除我们传参传的都是二级指针,因为我们都需要修改传过去的参数,也就是指向节点的指针。在之前的文章中,已经强调很多遍了,形参只是实参的一份临时拷贝,对形参的修改并不会影响实参。若传递结构体变量,是不会改变结构体变量本身的值的,如果我们需要在函数中改变一个变量的值,则需要传递这个值的地址,所以需要传址调用。

就比如头插SLTPushFront函数,我们需要将指向原先第一个节点的指针指向newnode,所以我们就要传第一个节点的地址的指针,也就是二级指针

其次,我们断言二级指针pphead不为空,目的是为了保证二级指针能够解引用得到一级指针*pphead,那有必要断言一级指针*pphead不为空吗?其实是没有必要的。如果*pphead为空,也就是没有任何节点的情况,其实就相当于申请新节点newnode,然后newnode->data为x,newnode->next为NULL。 

 • 🌰5.单链表尾部插入元素

向单链表的尾部插入元素,我们只需要创建一个新节点,让原先的单链表的最后一个节点的next指针指向这个新节点即可。

void SLTPushBack(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	SLTNode* newnode = SLTBuyNode(x);
	if (*pphead == NULL)
	{
		newnode = *pphead;
	}
	else
	{
		SLTNode* ptail = *pphead;
		while (ptail->next)
		{
			ptail = ptail->next;
		}
		ptail->next = newnode;
	}
}

单链表尾部插入元素,需要考虑*pphead为空的情况。头插不需要单独处理是因为及时*pphead为空,根据函数依然能创建一个节点,使得数据为x,next指针为NULL。而对于尾插而言,逻辑是让原来的最后一个节点的next指针指向newnode,而*pphead如果为NULL,也就是没有节点,那也就没有next指针,就没有办法处理这一情况,所以我们要单独处理。

如果*pphead为空,直接用SLTBuyNode申请一个新节点就可以了。对于*pphead不为空的情况,我们需要找到原单链表的最后一个节点。定义ptail指向第一个节点,利用while循环使得ptail指向原链表的最后一个节点,最后让ptail->next指针指向newnode即可。

 • 🌰6.指定位置之前插入数据

在pos节点的前面插入数据,需要先找到pos前一个的节点prev和创建的newnode节点,利用next指针将节点之间的关系设置好。

void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	assert(pphead && *pphead && pos);
	if (*pphead == pos)
	{
		SLTPushFront(pphead, x);
	}
	else
	{
		SLTNode* newnode = SLTBuyNode(x);;
		SLTNode* prev = *pphead;
		while (prev->next = pos)
		{
			prev = prev->next;
		}
		prev->next = newnode;
		newnode->next = pos;
	}
}

 在pos之前插入数据,需要得到三个节点:pos节点、pos前一个节点prev、newnode节点。由于需要得到pos前一个节点,所以当链表中只有一个节点的时候,我们没有办法找到,所以需要单独讨论,对于这种情况,其实就相当于头插SLTPushFront。

当链表中的节点大于1时,我们先创建newnode,然后利用while循环得到pos之前的prev节点,此时三个节点都有了,我们就可以进行操作了

 让prev的next指针指向newnode,再让newnode的next指针指向pos就可以了。

 • 🌰7.指定位置之后插入数据

在pos节点之后插入数据,我们需要得到pos节点、newnode节点和pos->next节点。

void SLTInsertAfter(SLTNode* pos, SLTDataType x)
{
	assert(pos);
	SLTNode* newnode = SLTBuyNode(x);;
	newnode->next = pos->next;
	pos->next = newnode;
}

我们观察下图,发现在pos后插入数据其实并不需要第一个节点,所以我们的节点参数传的只有pos。 

分析上图,有一个地方需要大家注意,就是我们必须先让newnode->next指向pos->next节点,而不能先让pos->next=newnode,也就是代码的最后两行不可以颠倒!这时因为我们如果先修改了pos->next,实际上pos->next=newnode并不是让pos->next指向newnode,根据上图不难思考,这一步实际上直接转移了newnode,新创建的节点就没有用了。总之,这两行代码的顺序是不可以调换的以后我们再处理复杂的节点关系,可以先从newnode入手,因为newnode是新的节点,和原链表没有直接的关系。

 • 🌰8.单链表头部删除元素

单链表头部删除元素需要将第一个节点的next指针保存下来,这样才能保证free释放后的第一个节点next指针可以访问。

void SLTPopFront(SLTNode** pphead)
{
	assert(pphead && *pphead);
	SLTNode* next = (*pphead)->next;
	free(*pphead);
	*pphead = next;
}

删除元素,一定要保证pphead和*pphead都不为NULL,其次,我们一定要在free之前就将(*pphead)->next指针保存下来,并且由于 * 的优先级低于 -> ,需要再*pphead上加上括号。

 注意,free只是放弃了对malloc开辟的空间的使用权限,*pphead还是可以继续使用的。

 • 🌰9.单链表尾部删除元素

尾删需要我们得到尾节点ptail和尾节点的前一个节点prev

void SLTPopBack(SLTNode** pphead)
{
	assert(pphead && *pphead);
	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		SLTNode* ptail = *pphead;
		SLTNode* prev = *pphead;
		while (ptail->next)
		{
			prev = ptail;
			ptail = ptail->next;
		}
		prev->next = NULL;
		free(ptail);
		ptail = NULL;
	}
}

一定要注意单链表中特殊情况需不需要分类讨论,标准就是如果特殊情况用普遍情况的代码也可以走得通就不需要单独讨论,反之则需要。

由于需要得到两个节点,当原链表中只有一个节点时,我们需要单独讨论。若只有一个节点, 我们直接释放掉它,再将它置为NULL就可以了。

 • 🌰10.删除指定位置的节点

删除指定的节点pos,需要得到pos节点、pos的前一个节点prev和pos->next节点。

void SLTErase(SLTNode** pphead, SLTNode* pos)
{
	assert(pphead && *pphead && pos);
	if (pos == *pphead)
	{
		SLTPopFront(pphead);
	}
	else
	{
		SLTNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = pos->next;
		free(pos);
		pos = NULL;
	}
}

 因为我们要得到prev节点,所以要对只有一个节点的情况分类讨论,这种情况直接用头删/尾删都是可以的。当节点数量大于1时,我们需要用while循环得到prev,再让prev->next指向pos->next就可以了。

此外,free释放后应该及时将pos置为NULL,这时一个好的习惯。 

 • 🌰11.删除指定位置之后的节点

删除pos的后一个节点,需要得到pos节点、pos->next节点和pos->next->next节点。

void SLTEraseAfter(SLTNode* pos)
{
	assert(pos && pos->next);
	SLTNode* del = pos->next;
	pos->next = del->next;
	free(del);
	del = NULL;
}

由题意不难发现,这样的链表必须至少有两个节点,所以assert需要保证pos节点和pos->next节点都不为空。

所以,我们不需要讨论特殊情况。由于pos->next指针如果free释放,就不能再通过它来访问pos->next->next,所以我们令del=pos->next,先del->next赋给pos->next,然后我们free释放del,这样就可以了。

 • 🌰12.单链表的查找

查找单链表中数据为x的节点。

SLTNode* SLTFind(SLTNode* phead, SLTDataType x)
{
	SLTNode* pcur = phead;
	while (pcur->next)
	{
		if (pcur->data == x)
			return pcur;
		pcur = pcur->next;
	}
	return NULL;
}

我们用pcur遍历整个链表,整个过程需要保证pcur的next指针不为NULL,因为如果为NULL,它就已经是最后一个节点了。当我们查找完整个链表都没有存储数据为x的节点时,我们直接返回NULL就可以了。

 • 🌰13.单链表的销毁

当我们操作完单链表完成我们需要做的事情后,需要对链表进行销毁。

void SListDesTroy(SLTNode** pphead)
{
	assert(pphead && *pphead);
	SLTNode* pcur = *pphead;
	while (pcur)
	{
		SLTNode* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	*pphead = NULL;
}

用pcur遍历整个链表,先用next指针保存下一个节点,然后释放当前节点,重复这个过程,最后再将第一个接地那也置为NULL就可以了。由于需要保证pcur能够访问到pcur->next,所以assert需要断言pphead和*pphead。

🍋知识点三:单链表基本操作

 单链表的基本操作就是实现对顺序表元素的增、删、查、改,关于如何实现已经在前面都进行了讲解,也给出了代码,现在希望大家掌握之后通过下面给出的SList.h头文件,在SList.c文件中分别实现这些功能,并在test.c的main函数中测试:

#pragma once

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>

typedef int SLTDataType;

typedef struct SListNode
{
	SLTDataType data;
	struct SListNode* next;
}SLTNode;

//链表的打印
void SLTPrint(SLTNode* phead);

//增加新节点
SLTNode* SLTBuyNode(SLTDataType x);

//尾插
void SLTPushBack(SLTNode** pphead, SLTDataType x);

//头插
void SLTPushFront(SLTNode** pphead, SLTDataType x);

//尾删
void SLTPopBack(SLTNode** pphead);

//头删
void SLTPopFront(SLTNode** pphead);

//查找
SLTNode* SLTFind(SLTNode* phead, SLTDataType x);

//在指定位置之前插入数据
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x);

//在指定位置之后插入数据
void SLTInsertAfter(SLTNode* pos, SLTDataType x);

//删除pos节点
void SLTErase(SLTNode** pphead, SLTNode* pos);

//删除pos之后的节点
void SLTEraseAfter(SLTNode* pos);

//销毁链表
void SListDesTroy(SLTNode** pphead);

• ✨SumUp结语

数据结构的学习一定要多画图,多理解,多思考,切忌直接抄写代码,就认为自己已经会了,一定到自己动手,才能明白自己哪个地方有问题。

 

如果大家觉得有帮助,麻烦大家点点赞,如果有错误的地方也欢迎大家指出~

 

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

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

相关文章

青动CRM源码搭建/部署/上线/运营/售后/更新

CRM是一款基于thinkphpfastadmin开发的客户管理系统。旨在助力企业销售全流程精细化、数字化管理&#xff0c;全面解决企业销售团队的全流程客户服务难题&#xff0c;帮助企业有效盘 活客户资源、量化销售行为&#xff0c;合理配置资源、建立科学销售体系&#xff0c;提升销售业…

无限免费泛域名SSL证书申请

如今https访问已经成为了网络安全的标识&#xff0c;SSL证书也成为了保护网站内用户信息安全和加密信息交互的手段之一。自2019年等保2.0的相应政策出台后&#xff0c;实现网站https访问也是必经环节之一。 当下SSL证书的相关政策也一直备受关注&#xff0c;有免费的SSL证书也…

【OV SSL证书】OV证书为什么更贵,又好在哪里?

证书类型有三种&#xff0c;分别是DV证书、OV证书、EV证书&#xff0c;本文将具体讲解OV证书的优点。 一、OV 证书的介绍 OV证书&#xff1a;企业验证型证书&#xff0c;证书审核方式为通过验证域名所有权和申请企业的真实身份信息才能签发证书。目前OV类型证书是全球运用最广…

Java的事件处理机制

Java事件处理机制 Java事件处理是采取“委派事件模型”。当事件发生时&#xff0c;产生事件的对象&#xff0c;会把此“信息”传递给“事件的监听者”处理&#xff0c;这里所说的“信息”实际上就是java.awt.event事件类库里某个类所创建的对象&#xff0c;把它称为“事件的对…

打包手动分包

手动分包 {ignore} 基本原理 手动分包的总体思路是&#xff1a; 先单独的打包公共模块 公共模块会被打包成为动态链接库(dll Dynamic Link Library)&#xff0c;并生成资源清单 根据入口模块进行正常打包 打包时&#xff0c;如果发现模块中使用了资源清单中描述的模块&…

基于 Llama-Index、Llama 3 和 Qdrant,构建一个 RAG 问答系统!

构建一个使用Llama-Index、Llama 3和Qdrant的高级重排-RAG系统 尽管大型语言模型&#xff08;LLMs&#xff09;有能力生成有意义且语法正确的文本&#xff0c;但它们面临的一个挑战是幻觉。 在LLMs中&#xff0c;幻觉指的是它们倾向于自信地生成错误答案&#xff0c;制造出看似…

人脸图像生成(DCGAN)

一、理论基础 1.什么是深度卷积对抗网络&#xff08;Deep Convolutional Generative Adversarial Network&#xff0c;&#xff09; 深度卷积对抗网络&#xff08;Deep Convolutional Generative Adversarial Network&#xff0c;DCGAN&#xff09;是一种生成对抗网络&#xf…

跨域问题(服务器和浏览器之间)待补充

一、为什么产生&#xff1a; 同源策略&#xff08;域名&#xff0c;协议&#xff0c;端口&#xff09;&#xff0c;安全问题 二、怎么解决&#xff1a; 1、cros:修改响应头 2、jp&#xff1a;采用js标签 3、代理&#xff08;创建服务器&#xff0c;定义规则&#xff0c;服…

十二届蓝桥杯Python组1月中/高级试题 第五题

** 十二届蓝桥杯Python组1月中/高级试题 第五题 ** 第五题&#xff08;难度系数 5&#xff0c;35 个计分点&#xff09; 提示信息&#xff1a; 平均数&#xff1a;是指在一组数据中所有数据之和再除以这组数据的个数。 如&#xff1a;“1&#xff0c;2&#xff0c;3&#xf…

安防监控/视频汇聚系统EasyCVR+AI智能分析助力解决校园霸凌事件

一、方案背景 校园霸凌这一校园中不应存在的现象&#xff0c;却屡见不鲜&#xff0c;它像一把锋利的刀&#xff0c;深深地刺入那些无辜的心灵&#xff0c;让受害者承受着无尽的痛苦。随着科技的进步与发展&#xff0c;我们应该追求有效、进步的手段来阻止校园霸凌事件的发生&a…

达坦科技@了你,并邀请你参加2024开源之夏!

开源之夏&#xff08;英文简称“OSPP”&#xff09;是中科院软件所“开源软件供应链点亮计划”指导下的系列暑期活动。达坦科技自开源之夏创办首期起每年参与&#xff0c;积极鼓励在校学生积极参与开源软件的开发维护&#xff0c;培养和发掘更多优秀的开发者。今年&#xff0c;…

超越机械抓手:看多指机器人如何灵活运用触觉?

论文标题&#xff1a; Learning Visuotactile Skills with Two Multifingered Hands 论文作者&#xff1a; Toru Lin, Yu Zhang, Qiyang Li, Haozhi Qi, Brent Yi, Sergey Levine, and Jitendra Malik 1. 机器人新挑战&#xff1a;多指手指操作 在自动化和智能化日益普及的…

【Vulhub靶场】Nginx 中间件漏洞复现

【Vulhub靶场】Nginx 中间件漏洞复现 一、Nginx 文件名逻辑漏洞&#xff08;CVE-2013-4547&#xff09;1. 影响版本2. 漏洞原理3. 漏洞复现 二、Nginx越界读取缓存漏洞&#xff08;CVE-2017-7529&#xff09;1. 漏洞详情2. 影响版本3. 漏洞复现 三、Nginx 配置错误导致漏洞&…

预告 | 飞凌嵌入式邀您共聚2024上海充换电展

第三届上海国际充电桩及换电站展览会&#xff08;CPSE&#xff09;&#xff0c;即将于5月22日~24日在上海汽车会展中心举行。届时&#xff0c;飞凌嵌入式将带来多款嵌入式核心板、开发板、充电桩TCU以及储能EMS网关产品&#xff0c;与来自全国的客户朋友及行业伙伴一同交流分享…

基于R语言绘图 | 转录代谢趋势图绘制教程

原文链接&#xff1a;基于R语言绘图 | 转录代谢趋势图绘制教程 本期教程 小杜的生信笔记&#xff0c;自2021年11月开始做的知识分享&#xff0c;主要内容是R语言绘图教程、转录组上游分析、转录组下游分析等内容。凡事在社群同学&#xff0c;可免费获得自2021年11月份至今全部…

【ArcGIS Pro微课1000例】0058:玩转NetCDF多维数据集

一、NetCDF介绍 NetCDF(network Common Data Form)网络通用数据格式是由美国大学大气研究协会(University Corporation for Atmospheric Research,UCAR)的Unidata项目科学家针对科学数据的特点开发的,是一种面向数组型并适于网络共享的数据的描述和编码标准。NetCDF广泛应…

羊大师:当代年轻人如何应对压力

羊大师&#xff1a;当代年轻人如何应对压力 当代年轻人面临各种压力&#xff0c;包括工作、学习、人际关系、经济等方面的压力。以下是一些建议&#xff0c;帮助年轻人应对这些压力&#xff1a; 认识并接受压力&#xff1a; 首先要认识到压力是生活中不可避免的一部分。 尝试…

WPF之DataGird应用

1&#xff0c;DataGrid相关属性 GridLinesVisibility&#xff1a;DataGrid网格线是否显示或者显示的方式。HorizontalGridLinesBrush&#xff1a;水平网格线画刷。VerticalGridLinesBrush&#xff1a;垂直网格线画刷。HorizontalScrollBarVisibility&#xff1a;水平滚动条可见…

卷积通用模型的剪枝、蒸馏---蒸馏篇--RKD关系蒸馏(以deeplabv3+为例)

本文使用RKD实现对deeplabv3+模型的蒸馏;与上一篇KD蒸馏的方法有所不同,RKD是对展平层的特征做蒸馏,蒸馏的loss分为二阶的距离损失Distance-wise Loss和三阶的角度损失Angle-wise Loss。 一、RKD简介 RKD算法的核心是以教师模型的多个输出为结构单元,取代传统蒸馏学习中以教…

【经验总结】 常用的模型优化器

优化器是一种用于优化模型权重和偏差的算法&#xff0c;它根据训练数据更新模型参数&#xff0c;以模型的预测结果更加准确。 1. 常见的优化器 SGD&#xff08;Stochastic Gradient Descent&#xff09;&#xff1a;SGD是一种基本的优化算法&#xff0c;它在每次迭代中随机选择…