【数据结构】- (带头结点)循环双向链表 - 详细实现思路及代码

news2025/1/9 1:24:43

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


在这里插入图片描述

一、概述

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

在这里插入图片描述

二、循环双向链表

循环双向链表:在单链表的每个结点中,再设置一个指向其前驱结点的指针域。且第一个结点(若有头结点,则是头结点)的前驱指针指向最后一个结点,最后一个结点的后继指针指向第一个结点(若有头结点,则是头结点)。
下图是 单链表
在这里插入图片描述
下图是 双向链表
在这里插入图片描述
下图是 循环双向链表
在这里插入图片描述

循环双向链表的特点:

  1. 循环双向链表可以反向访问到链表的结点,因为它有指向前一个结点的指针prior,也具备了循环链表的特点,可以在链表的任意位置访问其他结点;
  2. 带有头结点的循环双向链表,为空链表时,头结点的两个指针域都指向头结点
    此时,头指针list存放了头结点的地址,头结点的两个指针域也都存放了头结点的地址,所以会有list->next->next==list,不管list->后面接几个next,都会指向头结点。
    在这里插入图片描述
  3. 带有头结点的循环双向链表,为非空链表时,
    头结点的前驱指针域指向最后一个结点,后驱指针域指向第一个结点;
    最后一个结点的前驱指针域指向前一个结点,后驱指针域指向头结点
    其他结点的前驱指针域指向前一个结点,后驱指针域指向后一个结点;
    在这里插入图片描述

在这里插入图片描述

三、循环双向链表实现步骤

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

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

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

typedef int ElemType;
typedef struct _CyclicDoubleListNode
{
	ElemType data;
	struct _CyclicDoubleListNode *prior;	// 前驱指针
	struct _CyclicDoubleListNode *next;	// 后驱指针
}CyclicDoubleListNode;
typedef CyclicDoubleListNode* CyclicDoubleLinkList;

📌3.2 循环双向链表初始化

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

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

C语言实现代码如下:

CyclicDoubleLinkList ListInit()
{
	CyclicDoubleLinkList list = (CyclicDoubleLinkList)malloc(sizeof(CyclicDoubleListNode));
	list->prior = list;
	list->next = list;
	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(CyclicDoubleLinkList list, int data, int n)// 将node插入到第n位,n从1开始
{
	if(list==NULL || n<1) // 判断参数有效性
		return -1;
		
	CyclicDoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur->next!=list && cur_i<(n-1))
	{// 不是最后一个结点,且不是插入位置的前一个结点,就后移一个
		cur = cur->next;
		cur_i++;
	}
	if(cur->next==list)			// 移动到最后一个结点了
	{
		if(cur_i!=(n-1))		// 仍然不是插入位置前一个结点,出错
		{
			printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
			return -1;	// 链表没有 n 那么长
		}
	}
	CyclicDoubleListNode* new = (CyclicDoubleListNode*)malloc(sizeof(CyclicDoubleListNode));
	new->data = data;
	new->prior = cur;
	new->next = cur->next;
	cur->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==list),说明链表长度不够;
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(CyclicDoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	CyclicDoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur->next!=list && cur_i<(n-1))
	{// 不是最后一个结点,且当前位置不是删除位置的前一个,就后移一个
		cur = cur->next;
		cur_i++;
	}
	if(cur->next==list)		// 移动到最后一个结点了
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;		// 链表没有 n 那么长
	}
	CyclicDoubleListNode *delete = cur->next;
	delete->prior->next = delete->next;
	delete->next->prior = delete->prior;
	*data = delete->data;
	free(delete);
	return 0;
}

📌3.5 循环双向链表查找数据

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

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

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

C语言实现代码如下:

int ListFind(CyclicDoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	
	CyclicDoubleListNode* cur = list->next;// 指向第一个节点
	int cur_i=1;			// cur_i表示当前结点的序号
	if(cur == list)
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	while(cur->next!=list && cur_i<n)	
	{// 当前结点不是最后一个结点,且当前位置不是查找位置n,就往后移动一个
		cur = cur->next;
		cur_i++;
	}
	if(cur->next==list && cur_i!=n)	// 最后一个结点,且不是查找位置n
	{
		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、结束循环后,已经删除完所有节点,此时需要将头结点的两个指针域都指向头结点,表示空链表。

C语言实现代码如下:

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

在这里插入图片描述

四、循环双向链表完整代码

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

// CyclicDoubleLinkList.c
#include <stdio.h>
#include <stdlib.h>

typedef int ElemType;
typedef struct _CyclicDoubleListNode
{
	ElemType data;
	struct _CyclicDoubleListNode *prior;	// 前驱指针
	struct _CyclicDoubleListNode *next;	// 后驱指针
}CyclicDoubleListNode;
typedef CyclicDoubleListNode* CyclicDoubleLinkList;

CyclicDoubleLinkList ListInit()
{
	CyclicDoubleLinkList list = (CyclicDoubleLinkList)malloc(sizeof(CyclicDoubleListNode));
	list->prior = list;
	list->next = list;
	list->data = -1;
	return list;
}

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

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

int ListFind(CyclicDoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	
	CyclicDoubleListNode* cur = list->next;// 指向第一个节点
	int cur_i=1;			// cur_i表示当前结点的序号
	if(cur == list)
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	while(cur->next!=list && cur_i<n)	
	{// 当前结点不是最后一个结点,且当前位置不是查找位置n,就往后移动一个
		cur = cur->next;
		cur_i++;
	}
	if(cur->next==list && cur_i!=n)	// 最后一个结点,且不是查找位置n
	{
		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(CyclicDoubleLinkList list)
{
	CyclicDoubleListNode* cur = list->next;	// 指向第一个节点
	CyclicDoubleListNode* next = NULL;		// 用于保存下个结点地址
	while(cur->next != list)	// 当前不是最后一个,就往后移动
	{
		next = cur->next;		// 保存下个结点地址
		//printf("[%s %d]delete %d\n", __FUNCTION__,__LINE__, cur->data);
		free(cur);				// 删除当前结点、并释放内存
		cur = next;				// 将当前结点指针指向下个结点
	}
	list->prior = list;// 指向头结点
	list->next = list; // 指向头结点
}

void ListPrintf(CyclicDoubleLinkList list)
{
	CyclicDoubleListNode* cur = list->next;// 指向第一个节点
	printf("list:[");
	while(cur != list)
	{
		printf("%d,",cur->data);
		cur = cur->next;
	}
	printf("]\n");
}

int main()
{
	CyclicDoubleLinkList 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;
}

在这里插入图片描述

如果文章有帮助的话,点赞👍、收藏⭐,支持一波,谢谢 😁😁😁

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

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

相关文章

【python中的多线程了解一下?】

基本说明 线程&#xff08;Thread&#xff09;是操作系统进行调度的最小单位&#xff0c;是进程中的一个独立执行单元。线程与进程相比&#xff0c;具有更轻量级、更高效率、更易调度、共享资源等优点。 在传统的单核CPU中&#xff0c;操作系统通过时间片轮转算法将CPU的时间…

各种通讯总线的学习记要

一、在B站板道题看到一个比较好完视频&#xff08;爱上半导体&#xff09; 我觉得将232和485之前先将串口通信&#xff0c;因为它们都是串口通讯的变种。 串口通讯&#xff1a; 串口通讯我们约定好帧格式和波特率&#xff0c;通讯正常起始位为低开始&#xff0c;8位数据位&a…

【ABAQUS文档阅读笔记】关于体单元、壳单元、梁单元 、truss单元的总体认识

我的主页&#xff1a; 技术邻&#xff1a;小铭的ABAQUS学习的技术邻主页博客园 : HF_SO4的主页哔哩哔哩&#xff1a;小铭的ABAQUS学习的个人空间csdn&#xff1a;qgm1702 博客园文章链接&#xff1a; 学习笔记&#xff0c;from abaqus document “getting start with ABAQUS…

【代码随想录】刷题Day4

1.交换链表 24. 两两交换链表中的节点 前后指针实现 1.没有元素或者只有一个元素无意义 2.给出一个前驱prev&#xff0c;以及用来交换的两个节点cur和next 3.我当时是这么想的&#xff0c;如果两个指针一起动&#xff0c;那么就要用cur和next同时判断结束&#xff0c;也许这个…

C#基础学习--异常

目录 什么是异常 try语句 异常类 catch子句 catch子句段 finally块 为异常寻找处理程序 更进一步搜索 一般法则 抛出异常 不带异常对象的抛出 什么是异常 异常是程序运行时错误&#xff0c;它违反了系统约束或应用程序约束&#xff0c;或出现了在正常操作时未预料的情…

Windows环境下JDK内置迭代器的简介

我是荔园微风&#xff0c;作为一名在IT界整整25年的老兵&#xff0c;今天讨论一下Windows环境下JDK内置迭代器。 相信大家已经注意到&#xff0c;为了让广大程序员能够更加方便地操作聚合对象&#xff0c;在Java、C#等编程语言中都提供了内置迭代器。 如果对这种迭代器还不了…

在JetsonNano上安装PREEMPT-RT实时核+IghEtherCAT主站的详细步骤

特别说明&#xff1a;由于本人空闲时间有限&#xff0c;此博客中所有内容均是从本人的个人笔记中摘抄出来&#xff0c;命令行大多以图片形式展示&#xff0c;不能提供复制粘贴功能敬请体谅。此外&#xff0c;第一部分关于在 Jetson Nano 上安装PREEMPT-RT 实时核的步骤比较详细…

企业工商四要素核验 API:有效应对商业欺诈和恶意交易的利器

引言 企业工商四要素核验是一种用于验证企业基本信息的方法&#xff0c;主要包括企业名称、社会统一信用代码、法人名称、法人身份证四个要素。这些要素是企业注册时必须提供的信息&#xff0c;通过对这些信息的验证&#xff0c;可以确定企业的真实性和合法性&#xff0c;通常…

【力扣-21】合并两个有序链表

&#x1f58a;作者 : Djx_hmbb &#x1f4d8;专栏 : 数据结构 &#x1f606;今日分享 : 英文文章里 : TL;DR 或者 tl;dr。 网上英文解释有两种&#xff0c;一种是Too long;Don’t read&#xff0c;另一种是Too long;Didn’t read。意思是&#xff1a;“文章太长了&#xff0c;读…

Stable Diffusion MacBookAir 手把手安装教程,以及checkpoint安装、Lora安装、civitai介绍。

目标&#xff1a; 本地部署Stable Diffusion 安装步骤&#xff1a; 安装git Git - Downloadshttps://git-scm.com/downloads安装python 3.10.6 Python Releases for macOS | Python.orgThe official home of the Python Programming Languagehttps://www.python.org/downl…

再捐1亿元种树治沙:蚂蚁集团持续七年支持内蒙古生态治理

今天&#xff08;4月22日&#xff09;是“世界地球日”&#xff0c;内蒙古自治区林草局与蚂蚁集团启动战略合作&#xff1a;由蚂蚁集团在三年内再捐资1亿元&#xff0c;通过公益项目“蚂蚁森林”支持浑善达克沙地的生态治理。这1亿元将用于当地林草生态的修复保护、沙化土地的治…

助力电力行业数字化,BI 大有可为

各地密集上马的电力工程为上下游产业链发展带去了更多市场机遇&#xff0c;“三零”“三省”办电服务的推广让企业获得了看得见、摸得着的实惠&#xff0c;电力行业正在以自身高质量发展新成效助力经济稳增长&#xff0c;为扎实推进中国式现代化建设注入强劲动能。 电力行业是…

Spring Cloud Gateway Actuator API SpEL表达式注入命令执行(CVE-2022-22947)

Spring Cloud Gateway Actuator API SpEL表达式注入命令执行&#xff08;CVE-2022-22947&#xff09; 0x00 前言 Spring Cloud Gateway是Spring中的一个API网关。其3.1.0及3.0.6版本&#xff08;包含&#xff09;以前存在一处SpEL表达式注入漏洞&#xff0c;当攻击者可以访问…

plt.title()函数中文无法显示问题

文章目录 问题描述解决办法plt.title()函数参数说明 问题描述 由于画图时plt.title()默认是显示英文&#xff0c;如果我们设置标题为中文&#xff0c;会无法显示&#xff0c;如图&#xff1a; plt.title(训练损失) plt.plot(np.arange(len(losses)), losses, -o, colorred) p…

快速学会 IDEA 中 Git 的使用与 GitHub 仓库创建、连接、发布

&#x1f34e;道阻且长&#xff0c;行则将至。&#x1f353; 目录 一、Git &#x1f334;1.Git 工作流程2.Git 安装3.常用命令4.分支 二、远程仓库 GitHub &#x1f335;1.注册使用2.配置 SSH 公钥3.在 IDEA 中配置3.IDEA to GitHub4.创建分支5.commit 到仓库6.从远程仓库中抓…

【数据结构与算法】无队头指针的队列置空队、判队空 、入队和出队算法

题目 Qestion&#xff1a; 假设以带头结点的循环链表表示队列&#xff0c;并且只设一个指针指向队尾元素结点 (注意不设头指针) &#xff0c;试编写相应的置空队、判队空、入队和出队等算法。 核心思路 该队列的特殊之处&#xff1a; 用链表来表示队列该链表为带头节点的链表该…

VM ware workstation安装mac虚拟机

一. 工具准备 VMware workstation mac os的镜像文件&#xff08;链接&#xff1a;https://pan.baidu.com/s/1AdDGFU186bqKvW4wuGtpqw &#xff0c;提取码&#xff1a;tdvc&#xff09; 下载一个auto-unlocker文件&#xff08;https://github.com/paolo-projects/auto-unlocke…

人工智能十大流行算法,通俗易懂讲明白

人工智能是什么&#xff1f;很多人都知道&#xff0c;但大多又都说不清楚。 事实上&#xff0c;人工智能已经存在于我们生活中很久了。 比如我们常常用到的邮箱&#xff0c;其中垃圾邮件过滤就是依靠人工智能&#xff1b;比如每个智能手机都配备的指纹识别或人脸识别&#x…

「QT」QT中new的对象不需要delete的原因

博客主页:何曾参静谧的博客 文章专栏:「QT」QT5快速学习 目录 说明注意事项说明 在QT中,一般来说,通过new关键字创建的对象由QT的对象树来管理,而通过其他方式创建的对象就需要手动释放。QT的对象树可以看做是QT提供的一种内存管理机制,它能够自动地管理和释放对象的内存…

Xilinx之7系列时钟规划失败解析

目录 一 、前言 二、时钟规划 ​2.1 时钟单元经过非时钟路径 2.2 资源不足 2.3 跨半区 一 、前言 在设计的工程中&#xff0c;经常会在implementation的布局或布线阶段失败&#xff0c;下面将针对xilinx 7系列时钟规划失败的场景进行分类描述。本文示例器件为xc7k480tffv1…