当前位置: 首页 > news >正文

【数据结构与算法】 - 双向链表 - 详细实现思路及代码

目录
一、概述
二、双向链表
三、双向链表实现步骤
 📌3.1 C语言定义双向链表结点
 📌3.2 双向链表初始化
 📌3.3 双向链表插入数据
 📌3.4 双向链表删除数据
 📌3.5 双向链表查找数据
 📌3.6 双向链表的销毁
四、双向链表链表完整代码


在这里插入图片描述

一、概述

前几篇文章介绍了怎样去实现单链表、单循环链表,这篇文章主要介绍双向链表以及实现双向链表的步骤,最后提供我自己根据理解实现双向链表的C语言代码。跟着后面实现思路看下去,应该可以看懂代码,看懂代码后,就对双向链表有了比较抽象的理解了,最后自己再动手写一个双向链表,就基本理解这个东西了。
在这里插入图片描述

在这里插入图片描述

二、双向链表

双向链表:在单链表的每个结点中,再设置一个指向其前驱结点的指针域。
下图是 单链表
在这里插入图片描述

下图是 双向链表
在这里插入图片描述

双向链表的特点:

  1. 双向链表可以反向访问到链表的结点,因为它有指向前一个结点的指针prior
  2. 带有头结点的双向链表,为空链表时,头结点的两个指针域都指向NULL
    在这里插入图片描述
  3. 带有头结点的双向链表,为非空链表时,
    头结点的前驱指针域指向NULL,后驱指针域指向第一个结点;
    最后一个结点的前驱指针域指向前一个结点,后驱指针域指向NULL
    其他结点的前驱指针域指向前一个结点,后驱指针域指向后一个结点;
    在这里插入图片描述

在这里插入图片描述

三、双向链表实现步骤

从上面知道了双向链表的相关概念和一些特点,接下来开始实现双向链表,这里使用带有头结点的双向链表进行讲解,从初始化双向链表、插入数据、删除数据、查找数据、销毁双向链表5个操作进行说明,需要注意的是,双向链表的插入、删除操作需要改变两个指针域;其他操作基本和单链表一致。

📌3.1 C语言定义双向链表结点

为了和前几篇文章的链表做比较,双向链表结构体也尽量定义相似的。

typedef int ElemType;
typedef struct _DoubleListNode
{ElemType data;struct _DoubleListNode *prior;	// 前驱指针struct _DoubleListNode *next;	// 后驱指针
}DoubleListNode;
typedef DoubleListNode* DoubleLinkList;

📌3.2 双向链表初始化

因为带有头结点,初始化时就需要分配一个头结点的内存空间,且头指针会一直指向头结点。
双向链表初始化算法思路如下:

1、分配一个结点的存储空间作为头结点,并将头指针指向头结点;
2、让头结点的 prior指针 和 next指针 都指向NULL,头结点的数据填一个无效值;
3、将头指针返回给函数调用者。

C语言实现代码如下:

DoubleLinkList ListInit()
{DoubleLinkList list = (DoubleLinkList)malloc(sizeof(DoubleListNode));list->prior = NULL;list->next = NULL;list->data = -1;return list;
}

在这里插入图片描述

📌3.3 双向链表插入数据

双向链表插入数据大致分为两个步骤:首先,找到插入位置n的前一个结点;其次,是插入新结点,可以:先连接新结点、再指向新结点的顺序。
先连接新结点:是先把新结点的两个指针域分别连接当前结点和下个结点,new->prior = cur;new->next = cur->next;
再指向新结点:将当前节点的的指针域指向新节点,与旧节点断开,cur->next->prior = new;cur->next = new;
在这里插入图片描述

双向链表在第n个位置插入数据的算法思路:

1、定义一个结点指针cur指向头结点,用来遍历链表;
2、定义一个变量cur_i,用来表示当前结点的序号,初始化为0表示当前指向头结点;
3、将cur指针不断往后移动,直到下个位置就是插入位置n,即当cur_i==(n-1)跳出循环;
4、若结束循环后是当前结点无效,说明链表长度不够;
5、否则,说明当前结点cur的下个位置就是插入位置n,分配存储空间给新结点new;
6、把值填进新节点的数据域,用新结点prior指向当前结点,next指向当前节点的下个节点;
7、再将下个结点的prior指向新结点,当前结点的next指向新结点,完成插入操作。

C语言实现代码如下:

int ListInsert(DoubleLinkList list, int data, int n)// 将node插入到第n位,n从1开始
{if(list==NULL || n<1) // 判断参数有效性return -1;DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点int cur_i=0;			// cur_i表示当前结点的序号,0-头结点while(cur && cur_i<(n-1))// 当前结点有效,且不是插入位置的前一个结点,就后移一个{cur = cur->next;cur_i++;}if(!cur)			// 当前结点无效,说明已经移动到最后{printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);return -1;	// 链表没有 n 那么长}DoubleListNode* new = (DoubleListNode*)malloc(sizeof(DoubleListNode));new->data = data;new->prior = cur;new->next = cur->next;if(cur->next)		// 在最后一个结点插入时,cur->next==NULLcur->next->prior = new;cur->next = new;return 0;
}

📌3.4 双向链表删除数据

双向链表删除结点也是需要改变两个指针域,大致步骤如下,首先,找到删除位置n的前一个结点;其次,“把前一个结点的next指针域指向删除结点del的下个结点”,“再把下个结点的prior指针域指向删除结点del的前个结点”,这样就删除了下一个结点。
在这里插入图片描述

双向链表删除第n个数据的算法思路:

1、定义一个结点指针cur指向头结点,用来遍历链表;
2、定义一个变量cur_i,用来表示下个结点的序号,初始化为0表示当前指向头结点;
3、将cur指针不断往后移动,直到下个位置就是删除位置n,即当cur_i==(n-1)跳出循环;
4、若结束循环后是最后一个结点(cur->next==NULL),说明链表长度不够;
5、否则,说明下个结点(cur->next)就是删除位置n的结点delete,赋值delete = cur->next;
6、将前一个结点的next指针域指向 del 的下个结点 ,delete->prior->next = delete->next;
7、将下一个结点的prior指针域指向 del 的前个结点 ,delete->next->prior = delete->prior;;
8、最后释放delete结点的内存,完成删除操作。

C语言实现代码如下,删除结点更关注的是下个结点(cur->next)的有效性:

// 删除第n个结点,且将删除的值通过data传出
int ListDelete(DoubleLinkList list, int *data, int n)
{if(list==NULL || data==NULL || n<1)return -1;DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点int cur_i=0;				// cur_i表示当前结点的序号,0-头结点while(cur->next && cur_i<(n-1)){// 下个结点有效,且当前位置不是删除位置的前一个,就后移一个cur = cur->next;cur_i++;}if(!cur->next)		// 下个结点无效,说明已经移动到最后{printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);return -1;		// 链表没有 n 那么长}DoubleListNode *delete = cur->next;delete->prior->next = delete->next;delete->next->prior = delete->prior;free(delete);return 0;
}

📌3.5 双向链表查找数据

查找数据时,将指针指向第一个结点而非头结点,下面函数中list是头指针,指向头结点,双向链表非空时,list->next就是第一个结点;双向链表为空时,list->next == NULL。双向链表 和 单链表 查找数据的算法是一样的。

双向链表查找第n个数据的算法思路:

1、定义一个结点指针cur指向第一个结点(list->next),用来遍历链表;
2、定义一个变量cur_i,用来表示当前结点的序号,初始化为1(第一步指向的就是第一个结点);
3、若当前结点有效,且当前位置不是查找位置n,就继续后移,直到最后结点或cur_i==n跳出循环;
4、若结束循环后,当前结点无效,说明已经移动到最后,链表长度不够;
5、否则,说明当前结点(cur)就是查找位置n的结点;返回结点数据*data = cur->data。

C语言实现代码如下:

int ListFind(DoubleLinkList list, int *data, int n)
{if(list==NULL || data==NULL || n<1)return -1;DoubleListNode* cur = list->next;// 指向第一个节点int cur_i=1;			// i表示当前结点的序号while(cur && cur_i<n)	// 当前结点有效,且当前位置不是查找位置n,就往后移动一个{cur = cur->next;cur_i++;}if(!cur)			// 当前结点无效,说明已经移动到最后{printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);return -1;	// 链表没有 n 那么长}*data = cur->data;printf("[%s %d]find No.%d = %d\n", __FUNCTION__,__LINE__, n,*data);return 0;
}

📌3.6 双向链表的销毁

双向链表销毁的算法思路:

1、定义一个结点指针cur指向第一个结点,用来遍历链表;
2、定义一个结点指针next,保存下个结点地址;
3、当前指针不是指向最后一个结点的指针域就后移,进入循环:3.1、先保存下个结点地址,因为下个结点本来保存在cur->next,直接free(cur)会丢掉下个结点;3.2、删除当前结点,释放内存3.3、将当前指针指向前面保存好的下个结点。
4、结束循环后,已经删除完所有节点,此时需要将头结点的两个指针域都指向NULL,表示空链表。

C语言实现代码如下:

void ListDestroy(DoubleLinkList list)
{DoubleListNode* cur = list->next;	// 指向第一个节点DoubleListNode* next = NULL;		// 用于保存下个结点地址while(cur)	// 当前结点有效,就往后移动{next = cur->next;		// 保存下个结点地址//printf("[%s %d]delete %d\n", __FUNCTION__,__LINE__, cur->data);free(cur);				// 删除当前结点、并释放内存cur = next;				// 将当前结点指针指向下个结点}list->prior = NULL;list->next = NULL;
}

在这里插入图片描述

四、双向链表完整代码

代码只是为了更好地了解循环链表,实现过程可能存在不足,有发现的,欢迎指正,谢谢!!!
代码已在Ubuntu编译通过,可执行。

// DoubleList.c
#include <stdio.h>
#include <stdlib.h>typedef int ElemType;
typedef struct _DoubleListNode
{ElemType data;struct _DoubleListNode *prior;	// 前驱指针struct _DoubleListNode *next;	// 后驱指针
}DoubleListNode;
typedef DoubleListNode* DoubleLinkList;DoubleLinkList ListInit()
{DoubleLinkList list = (DoubleLinkList)malloc(sizeof(DoubleListNode));list->prior = NULL;list->next = NULL;list->data = -1;return list;
}int ListInsert(DoubleLinkList list, int data, int n)// 将node插入到第n位,n从1开始
{if(list==NULL || n<1) // 判断参数有效性return -1;DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点int cur_i=0;				// cur_i表示当前结点的序号,0-头结点while(cur && cur_i<(n-1))// 当前结点有效,且不是插入位置的前一个结点,就后移一个{cur = cur->next;cur_i++;}if(!cur)			// 当前结点无效,说明已经移动到最后{printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);return -1;	// 链表没有 n 那么长}DoubleListNode* new = (DoubleListNode*)malloc(sizeof(DoubleListNode));new->data = data;new->prior = cur;new->next = cur->next;if(cur->next)		// 在最后一个结点插入时,cur->next==NULLcur->next->prior = new;cur->next = new;return 0;
}// 删除第n个结点,且将删除的值通过data传出
int ListDelete(DoubleLinkList list, int *data, int n)
{if(list==NULL || data==NULL || n<1)return -1;DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点int cur_i=0;				// cur_i表示当前结点的序号,0-头结点while(cur->next && cur_i<(n-1)){// 下个结点有效,且当前位置不是删除位置的前一个,就后移一个cur = cur->next;cur_i++;}if(!cur->next)		// 下个结点无效,说明已经移动到最后{printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);return -1;		// 链表没有 n 那么长}DoubleListNode *delete = cur->next;delete->prior->next = delete->next;delete->next->prior = delete->prior;free(delete);return 0;
}int ListFind(DoubleLinkList list, int *data, int n)
{if(list==NULL || data==NULL || n<1)return -1;DoubleListNode* cur = list->next;// 指向第一个节点int cur_i=1;			// i表示当前结点的序号while(cur && cur_i<n)	// 当前结点有效,且当前位置不是查找位置n,就往后移动一个{cur = cur->next;cur_i++;}if(!cur)			// 当前结点无效,说明已经移动到最后{printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);return -1;	// 链表没有 n 那么长}*data = cur->data;printf("[%s %d]find No.%d = %d\n", __FUNCTION__,__LINE__, n,*data);return 0;
}void ListDestroy(DoubleLinkList list)
{DoubleListNode* cur = list->next;	// 指向第一个节点DoubleListNode* next = NULL;		// 用于保存下个结点地址while(cur)	// 当前结点有效,就往后移动{next = cur->next;		// 保存下个结点地址//printf("[%s %d]delete %d\n", __FUNCTION__,__LINE__, cur->data);free(cur);				// 删除当前结点、并释放内存cur = next;				// 将当前结点指针指向下个结点}list->prior = NULL;list->next = NULL;
}void ListPrintf(DoubleLinkList list)
{DoubleListNode* cur = list->next;// 指向第一个节点printf("list:[");while(cur){printf("%d,",cur->data);cur = cur->next;}printf("]\n");
}int main()
{DoubleLinkList list=ListInit();int data=0;printf("Linklist is empty !!! \n");ListInsert(list, 2, 2);		// 空链表时,验证插入ListDelete(list, &data, 1);	// 空链表时,验证删除ListFind(list, &data, 1);	// 空链表时,验证查询ListDestroy(list);			// 空链表时,验证销毁printf("\ninsert 3 data\n");// 正常插入3个数据ListInsert(list, 1, 1);ListInsert(list, 2, 2);ListInsert(list, 3, 3);ListPrintf(list);printf("\n验证错误值\n");ListInsert(list, 5, 5);		// 验证插入ListDelete(list, &data, 4);	// 验证删除ListFind(list, &data, 4);	// 验证查询printf("\n正常操作\n");// 正常操作ListFind(list, &data, 2);printf("delete 2,now\n");ListDelete(list, &data, 2);ListPrintf(list);printf("Insert 4 to 2,now\n");ListInsert(list, 4, 2);ListPrintf(list);printf("Destroy ,now\n");ListDestroy(list);ListPrintf(list);return 0;
}

相关文章:

【数据结构与算法】 - 双向链表 - 详细实现思路及代码

目录 一、概述 二、双向链表 三、双向链表实现步骤  &#x1f4cc;3.1 C语言定义双向链表结点  &#x1f4cc;3.2 双向链表初始化  &#x1f4cc;3.3 双向链表插入数据  &#x1f4cc;3.4 双向链表删除数据  &#x1f4cc;3.5 双向链表查找数据  &#x1f4cc;3.6 双向链…...

面试官在线点评4份留学生简历! 这些坑你中了几个?如何写项目描述才能被大厂发面试?转专业简历该咋写 | 还有优秀简历展示!

我们给大家展示一下 从材料的准备 也就是说到底包含哪些具体的项目 为什么说这些项目是不错的 第二呢就是说在陈述上 在整个这个简历的结构 他的完备性他的准确性 他的正确性 以及最后他的具体的这种项目的描述 那讲完了这个好的简历呢 我们另外搜集了几份简历 那这些简历呢其实…...

一觉醒后ChatGPT 被淘汰了

OpenAI 的 Andrej Karpathy 都大力宣传&#xff0c;认为 AutoGPT 是 prompt 工程的下一个前沿。 近日&#xff0c;AI 界貌似出现了一种新的趋势&#xff1a;自主人工智能。 这不是空穴来风&#xff0c;最近一个名为 AutoGPT 的研究开始走进大众视野。特斯拉前 AI 总监、刚刚回归…...

spring框架的事务

1.什么是事务? 事务&#xff1a;是数据库操作的最小工作单元&#xff0c;是作为单个逻辑工作单元执行的一系列操作&#xff1b;这些操作作为一个整体一起向系统提交&#xff0c;要么都执行、要么都不执行&#xff1b;事务是一组不可再分割的操作集合&#xff08;工作逻辑单元…...

Spring配置数据源

Spring配置数据源数据源的作用环境准备手动创建c3p0数据源封装抽取关键信息&#xff0c;手动创建c3p0数据源使用Spring容器配置数据源数据源的作用 数据源(连接池)是提高程序性能如出现的 事先实例化数据源&#xff0c;初始化部分连接资源 使用连接资源时从数据源中获取 使用完…...

【前端之旅】Vue入门笔记

一名软件工程专业学生的前端之旅,记录自己对三件套(HTML、CSS、JavaScript)、Jquery、Ajax、Axios、Bootstrap、Node.js、Vue、小程序开发(Uniapp)以及各种UI组件库、前端框架的学习。 【前端之旅】Web基础与开发工具 【前端之旅】手把手教你安装VS Code并附上超实用插件…...

WPF教程(二)--Application WPF程序启动方式

1.Application介绍 WPF与WinForm一样有一个 Application对象来进行一些全局的行为和操作&#xff0c;并且每个 Domain &#xff08;应用程序域&#xff09;中仅且只有一个 Application 实例存在。和 WinForm 不同的是WPF Application默认由两部分组成 : App.xaml 和 App.xaml.…...

snmp 自定义子代理mib库

测试环境&#xff1a;centos8 1、安装软件 yum install -y net-snmp net-snmp-utils yum install -y net-snmp-perl net-snmp-devel net-snmp-libs 2、创建用户 net-snmp-create-v3-user 输入用户名 soft 输入密码 123456 输入密码 654321 service snmpd restart 3、创建…...

一文说透安全沙箱技术

在数字经济的东风中&#xff0c;数据安全至关重要。目前已经颁布了包括《数据安全法》、《个人信息保护法》和《数据安全管理办法》在内的国家政策&#xff0c;以促进整个数据要素的发展。 而近年来&#xff0c;随着移动应用程序的普及和小程序技术的崛起&#xff0c;安全沙箱…...

Java多线程基础面试总结(二)

创建三种线程的方式对比 使用实现Runnable、Callable接口的方式创建多线程。 优势 Java的设计是单继承的设计&#xff0c;如果使用继承Thread的方式实现多线程&#xff0c;则不能继承其他的类&#xff0c;而如果使用实现Runnable接口或Callable接口的方式实现多线程&#xf…...

NS32F407VGT6 NS32F407VET6软硬件通用STM32F407VGT6 407VET6

NS32F407VGT6 NS32F407VET6 器件基于高性能的 ARM Cortex-M4 32 位 RISC 内核&#xff0c;工作频率高达 168MHz 。 Cortex-M4 内核带有单精度浮点运算单元 (FPU) &#xff0c;支持所有 ARM 单精度数据处理指令和数据类型。它还 具有一组 DSP 指令和提高应用安全性的一…...

Openstack: network: ovs: dpif/show 实例分析:interface

[TOC 实例 [cbis-adminovercloud–13 (overcloudrc) ~]$ sudo ovs-appctl dpif/show systemovs-system: hit:75198007884 missed:109924265 br-ex: br-ex 65534/3: (internal) ,65534 是port number; OpenFlow port number&#xff1b; 3 是 ofp_port_to_odp_port(ofproto, o…...

必要的项目管理软件因素

什么样的项目管理软件好&#xff1f;对于一个项目团队来说&#xff0c;从项目开始到项目结束&#xff0c;需要多个部门的配合。每个成员可能会参与一个以上的项目&#xff0c;这通常需要并行的多个项目。据介绍&#xff0c;国外90%以上的项目是用软件管理的&#xff0c;而中国只…...

大学刚毕业,用10000小时,走进字节跳动拿了offer

前言&#xff1a; 没有绝对的天才&#xff0c;只有持续不断的付出。对于我们每一个平凡人来说&#xff0c;改变命运只能依靠努力幸运&#xff0c;但如果你不够幸运&#xff0c;那就只能拉高努力的占比。 2020年7月&#xff0c;我有幸成为了字节跳动的一名测试开发&#xff0c…...

docker 安装 redis

搜索镜像 docker search redis 拉取最新版本 Docker pull redis Docker挂载配置文件 docker run --restartalways --log-opt max-size100m --log-opt max-file2 -p 6379:6379 --name myredis -v /opt/myredis/redis.conf:/etc/redis/redis.conf -v /opt/myredis/data:/d…...

Ceph常见问题

1. CephFS问题诊断 1.1 无法创建 创建新CephFS报错Error EINVAL: pool ‘rbd-ssd’ already contains some objects. Use an empty pool instead&#xff0c;解决办法&#xff1a; ceph fs new cephfs rbd-ssd rbd-hdd --force1.2 mds.0 is damaged 断电后出现此问题。MDS进…...

Android---Jetpack之Paging

目录 Paging 组件的意思 Paging 支持的架构类型 Paging 的工作原理 PositionalDataSource PagekeyedDataSource ItemKeyedDataSource BoundaryCallback Paging 组件的意思 分页加载是在应用程序开发过程中十分常见的需求&#xff0c;Paging 就是 Google 为了方便 Andr…...

gensim.models.word2vec() 参数详解

1. Word2vec简介 Word2vec是一个用来产生词向量的模型。是一个将单词转换成向量形式的工具。 通过转换&#xff0c;可以把对文本内容的处理简化为向量空间中的向量运算&#xff0c;计算出向量空间上的相似度&#xff0c;来表示文本语义上的相似度。 2.Word2vec参数详解 class…...

光栅和矢量图像处理SDK:Graphics Mill 11.7Crack

Graphics Mill 是适用于 .NET 和 ASP.NET 开发人员的最强大的成像工具集。它允许用户轻松地向 .NET 应用程序添加复杂的光栅和矢量图像处理功能。 光栅图形 加载和保存 JPEG、PNG PSD 和其他 8 种图像格式 调整大小、裁剪、自动修复、色度键和 30 多种其他图像处理 使用任何维度…...

阿里云的客服 锻炼你心性的 一种方式 !!!

阿里云的产品&#xff0c;非常棒&#xff0c;开发的同学非常棒&#xff0c;专家们更棒&#xff0c;但&#xff0c;一切的开始就怕一个但字&#xff0c;但我还的说&#xff0c;但&#xff0c;阿里云的客服&#xff0c;OMG &#xff0c;我已经忍耐了 1年了&#xff0c;是在忍不住…...

(十)学生端搭建

本次旨在将之前的已完成的部分功能进行拼装到学生端&#xff0c;同时完善学生端的构建。本次工作主要包括&#xff1a; 1.学生端整体界面布局 2.模拟考场与部分个人画像流程的串联 3.整体学生端逻辑 一、学生端 在主界面可以选择自己的用户角色 选择学生则进入学生登录界面…...

《从零掌握MIPI CSI-2: 协议精解与FPGA摄像头开发实战》-- CSI-2 协议详细解析 (一)

CSI-2 协议详细解析 (一&#xff09; 1. CSI-2层定义&#xff08;CSI-2 Layer Definitions&#xff09; 分层结构 &#xff1a;CSI-2协议分为6层&#xff1a; 物理层&#xff08;PHY Layer&#xff09; &#xff1a; 定义电气特性、时钟机制和传输介质&#xff08;导线&#…...

【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表

1、行为树节点分类 在 Nav2(Navigation2)的行为树框架中,行为树节点插件按照功能分为 Action(动作节点)、Condition(条件节点)、Control(控制节点) 和 Decorator(装饰节点) 四类。 1.1 动作节点 Action 执行具体的机器人操作或任务,直接与硬件、传感器或外部系统…...

Spring AI 入门:Java 开发者的生成式 AI 实践之路

一、Spring AI 简介 在人工智能技术快速迭代的今天&#xff0c;Spring AI 作为 Spring 生态系统的新生力量&#xff0c;正在成为 Java 开发者拥抱生成式 AI 的最佳选择。该框架通过模块化设计实现了与主流 AI 服务&#xff08;如 OpenAI、Anthropic&#xff09;的无缝对接&…...

大学生职业发展与就业创业指导教学评价

这里是引用 作为软工2203/2204班的学生&#xff0c;我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要&#xff0c;而您认真负责的教学态度&#xff0c;让课程的每一部分都充满了实用价值。 尤其让我…...

Java多线程实现之Thread类深度解析

Java多线程实现之Thread类深度解析 一、多线程基础概念1.1 什么是线程1.2 多线程的优势1.3 Java多线程模型 二、Thread类的基本结构与构造函数2.1 Thread类的继承关系2.2 构造函数 三、创建和启动线程3.1 继承Thread类创建线程3.2 实现Runnable接口创建线程 四、Thread类的核心…...

【Redis】笔记|第8节|大厂高并发缓存架构实战与优化

缓存架构 代码结构 代码详情 功能点&#xff1a; 多级缓存&#xff0c;先查本地缓存&#xff0c;再查Redis&#xff0c;最后才查数据库热点数据重建逻辑使用分布式锁&#xff0c;二次查询更新缓存采用读写锁提升性能采用Redis的发布订阅机制通知所有实例更新本地缓存适用读多…...

基于IDIG-GAN的小样本电机轴承故障诊断

目录 🔍 核心问题 一、IDIG-GAN模型原理 1. 整体架构 2. 核心创新点 (1) ​梯度归一化(Gradient Normalization)​​ (2) ​判别器梯度间隙正则化(Discriminator Gradient Gap Regularization)​​ (3) ​自注意力机制(Self-Attention)​​ 3. 完整损失函数 二…...

PH热榜 | 2025-06-08

1. Thiings 标语&#xff1a;一套超过1900个免费AI生成的3D图标集合 介绍&#xff1a;Thiings是一个不断扩展的免费AI生成3D图标库&#xff0c;目前已有超过1900个图标。你可以按照主题浏览&#xff0c;生成自己的图标&#xff0c;或者下载整个图标集。所有图标都可以在个人或…...

Java数组Arrays操作全攻略

Arrays类的概述 Java中的Arrays类位于java.util包中&#xff0c;提供了一系列静态方法用于操作数组&#xff08;如排序、搜索、填充、比较等&#xff09;。这些方法适用于基本类型数组和对象数组。 常用成员方法及代码示例 排序&#xff08;sort&#xff09; 对数组进行升序…...