【数据结构】带头双向循环链表---C语言版(单链表我们分手吧,不要再找我玩了!!!)

在这里插入图片描述

文章目录

  • 🐸一、前言
  • 🐸二、链表的分类
    • 🍄1. 单向或者双向链表
    • 🍄2. 带头或者不带头链表
    • 🍄3. 循环或者非循环
    • 🍄4. 最常用链表
  • 🐸三、带头双向循环链表详解
    • 🍎创建带头双向循环链表
    • ⭕接口1:定义结构体(LTNode)
    • ⭕接口2:初始化(创建哨兵卫)(LTInit)
    • ⭕接口3:打印(LTPrint)
    • ⭕接口4:创建新结点(BuyLTNode)
    • ⭕接口5:释放(LTDestroy)
    • ⭕接口6:判空(LTEmpty)
    • ⭕接口7:头插(LTPushFront)
    • ⭕接口8:尾插(LTPushBack)
    • ⭕接口9:头删(LTPopFront)
    • ⭕接口10:尾删(LTPopBack)
    • ⭕接口11:查找(LTFind)
    • ⭕接口12:修改(LTModify)
    • ⭕接口13:在pos之前插入(LTInsert)
    • ⭕接口14:删除pos位置的值(LTErase)
  • 🐸四、完整代码
    • 🥝List.h
    • 🥝List.c
    • 🥝Test.c

在这里插入图片描述

🐸一、前言

在前面我们学习实现了单链表(无头单向不循环链表),这里我们引入带头双向循环链表
很明显这两种链表的结构截然不同,但都是作为链表最常使用链表结构
前者因其结构上的缺点而作为面试考题的常驻嘉宾,而且复杂麻烦
后者则是以结构最优著称,实现起来也是非常的简单(少了单链表头节点,尾节点,前一节点等问题的困扰),可以说是最的链表结构🤭

🐸二、链表的分类

🍄1. 单向或者双向链表

在这里插入图片描述

  • 单向:节点结构中只存在下一节点的地址,所以难以从后一节点找到前一节点
  • 双向:节点结构中存在前一节点和后一节点的地址,寻找前一节点和后一节点很便利

🍄2. 带头或者不带头链表

在这里插入图片描述

  • 带头:在本来的头节点之前还有一个哨兵卫节点作为头节点,它的址域指针指向头节点,值域不做使用
  • 不带头:没有哨兵卫头节点,在尾删尾插等问题中要考虑头节点的情况(局限)

🍄3. 循环或者非循环

在这里插入图片描述

  • 循环:头节点会与尾节点相连
  • 非循环:头节点不与尾节点相连

🍄4. 最常用链表

虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:
🔴无头单向非循环链表
在这里插入图片描述
🔴带头双向循环链表
在这里插入图片描述

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

🐸三、带头双向循环链表详解

在这里插入图片描述

🍎创建带头双向循环链表

🥰这里先创建三个文件:
1️⃣:List.h文件用于函数的声明
2️⃣:List.c文件用于函数的定义
3️⃣:Test.c文件用于测试函数
建立三个文件的目的: 将链表作为一个项目来进行编写,方便我们的学习与观察。

⭕接口1:定义结构体(LTNode)

🥰请看代码与注释👇

//自定义类型
typedef int ListNodeDataType;

//创建双向链表
typedef struct ListNode
{
	struct ListNode* prev; //前址域
	struct ListNode* next; //后址域
	ListNodeDataType data; //值域
}LTNode;

⭕接口2:初始化(创建哨兵卫)(LTInit)

🥰请看代码与注释👇

//初始化(创建哨兵卫)
LTNode* LTInit()
{
	LTNode* phead = BuyLTNode(-1); //哨兵卫不存储有效值
	phead->prev = phead; //初始化哨兵卫头节点址域
	phead->next = phead;

	return phead;
}

⭕接口3:打印(LTPrint)

🥰请看代码与注释👇

//打印
void LTPrint(LTNode* phead)
{
	//断言传入指针不为NULL
	assert(phead);

	printf("guard<==>");

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		printf("%d<==>", cur->data); //打印数据
		cur = cur->next; //找到下一个节点
	}
	printf("\n");
}

⭕接口4:创建新结点(BuyLTNode)

🥰请看代码与注释👇

//创建新节点
LTNode* BuyLTNode(ListNodeDataType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	if (newnode == NULL)
	{
		perror("malloc fail"); //失败打印错误信息并结束进程
		return;
	}
	//初始化节点
	newnode->data = x;
	newnode->prev = NULL;
	newnode->next = NULL;

	return newnode;
}

⭕接口5:释放(LTDestroy)

🥰请看代码与注释👇

//释放
void LTDestroy(LTNode* phead)
{
	//断言传入指针不为NULL
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		LTNode* next = cur->next; //记录下一个节点地址
		free(cur); //释放当前节点
		cur = next; //找到下一个节点
	}

	free(phead);
}

⭕接口6:判空(LTEmpty)

🥰请看代码与注释👇

//判空
bool LTEmpty(LTNode* phead)
{
	assert(phead);

	return phead->next == phead; //判断只剩哨兵卫头结点的情况
}

⭕接口7:头插(LTPushFront)

🥰请看代码与注释👇

//头插
void LTPushFront(LTNode* phead, ListNodeDataType x)
{
	assert(phead);

	LTNode* newnode = BuyLTNode(x);
	LTNode* first = phead->next; //记录哨兵卫头结点的下一节点
	
	//构建各节点之间的关系
	phead->next = newnode;
	newnode->prev = phead;

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

⭕接口8:尾插(LTPushBack)

🥰请看代码与注释👇

//尾插
void LTPushBack(LTNode* phead, ListNodeDataType x)
{
	assert(phead);

	LTNode* tail = phead->prev; //找到尾节点
	LTNode* newnode = BuyLTNode(x);
	
	//构建尾节点与新节点,新节点与哨兵卫头结点的关系
	tail->next = newnode;
	newnode->prev = tail;

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

⭕接口9:头删(LTPopFront)

🥰请看代码与注释👇

//头删
void LTPopFront(LTNode* phead)
{
	assert(!LTEmpty(phead));

	LTNode* first = phead->next; //记录哨兵卫头节点下一节点及其的下一节点
	LTNode* second = first->next;

	phead->next = second;
	second->prev = phead;
	free(first);
}

⭕接口10:尾删(LTPopBack)

🥰请看代码与注释👇

//尾删
void LTPopBack(LTNode* phead)
{
	//判空 以及 判断只剩哨兵卫头结点的情况
	assert(!LTEmpty(phead));
	
	//记录尾节点及其前一节点
	LTNode* tail = phead->prev;
	LTNode* tailPrev = tail->prev;
	
	//构建尾节点前一节点与哨兵卫头结点的关系
	tailPrev->next = phead;
	phead->prev = tailPrev;
	free(tail); //释放尾节点
}

⭕接口11:查找(LTFind)

🥰请看代码与注释👇

LTNode* LTFind(LTNode* phead, ListNodeDataType x)
{
	assert(phead);

	LTNode* cur = phead->next;;
	while (cur != phead)
	{
		if (cur->data == x) //比较数据
		{
			return cur;
		}
		
		cur = cur->next; //找到下一个节点
	}

	return NULL; //没找到则返回NULL
}

⭕接口12:修改(LTModify)

🥰请看代码与注释👇

//修改
void LTModify(LTNode* phead, LTNode* pos, ListNodeDataType x)
{
	assert(phead);
	assert(pos);

	pos->data = x;
}

⭕接口13:在pos之前插入(LTInsert)

🥰请看代码与注释👇

//在pos之前插入
void LTInsert(LTNode* pos, ListNodeDataType x)
{
	assert(pos);

	LTNode* prev = pos->prev;
	LTNode* newnode = BuyLTNode(x);
	
	prev->next = newnode;
	newnode->prev = prev;
	
	newnode->next = pos;
	pos->prev = newnode;
}

⭕接口14:删除pos位置的值(LTErase)

🥰请看代码与注释👇

//删除pos位置的值
void LTErase(LTNode* pos)
{
	assert(pos);
	
	//记录pos的前一节点和后一节点
	LTNode* posPrev = pos->prev;
	LTNode* posNext = pos->next;

	posPrev->next = posNext;
	posNext->prev = posPrev;
	free(pos); //释放节点
}

🐸四、完整代码

🥝List.h

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

//自定义类型
typedef int ListNodeDataType;

//创建双向链表
typedef struct ListNode
{
	struct ListNode* prev;
	struct ListNode* next;
	ListNodeDataType data;
}LTNode;

//初始化(创建哨兵卫)
LTNode* LTInit();
//打印
void LTPrint(LTNode* phead);
//释放
void LTDestroy(LTNode* phead);
//判空
bool LTEmpty(LTNode* phead);
//头插
void LTPushFront(LTNode* phead, ListNodeDataType x);
//尾插
void LTPushBack(LTNode* phead, ListNodeDataType x);
//头删
void LTPopFront(LTNode* phead);
//尾删
void LTPopBack(LTNode* phead);
//查找
LTNode* LTFind(LTNode* phead, ListNodeDataType x);
//修改
void LTModify(LTNode* phead, LTNode* pos, ListNodeDataType x);
//在pos之前插入
void LTInsert(LTNode* pos, ListNodeDataType x);
//删除pos位置的值
void LTErase(LTNode* pos);

🥝List.c

#include "List.h"

//创建新节点
LTNode* BuyLTNode(ListNodeDataType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	if (newnode == NULL)
	{
		perror("malloc fail");
		return;
	}

	newnode->data = x;
	newnode->prev = NULL;
	newnode->next = NULL;

	return newnode;
}

//初始化(创建哨兵卫)
LTNode* LTInit()
{
	LTNode* phead = BuyLTNode(-1); //哨兵卫不存储有效值
	phead->prev = phead;
	phead->next = phead;

	return phead;
}

//打印
void LTPrint(LTNode* phead)
{
	assert(phead);

	printf("guard<==>");

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		printf("%d<==>", cur->data);
		cur = cur->next;
	}
	printf("\n");
}

//释放
void LTDestroy(LTNode* phead)
{
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		LTNode* next = cur->next;
		free(cur);
		cur = next;
	}

	free(phead);
}

//判空
bool LTEmpty(LTNode* phead)
{
	assert(phead);

	return phead->next == phead;
}

//头插
void LTPushFront(LTNode* phead, ListNodeDataType x)
{
	assert(phead);

	LTNode* newnode = BuyLTNode(x);
	LTNode* first = phead->next;

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

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

头插
//void LTPushFront(LTNode* phead, ListNodeDataType x)
//{
//	assert(phead);
//
//	LTInsert(phead->next, x);
//}

//尾插
void LTPushBack(LTNode* phead, ListNodeDataType x)
{
	assert(phead);

	LTNode* tail = phead->prev;
	LTNode* newnode = BuyLTNode(x);
	
	tail->next = newnode;
	newnode->prev = tail;

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

尾插
//void LTPushBack(LTNode* phead, ListNodeDataType x)
//{
//	assert(phead);
//
//	LTInsert(phead, x);
//}

//头删
void LTPopFront(LTNode* phead)
{
	assert(!LTEmpty(phead));

	LTNode* first = phead->next;
	LTNode* second = first->next;

	phead->next = second;
	second->prev = phead;
	free(first);
}

头删
//void LTPopFront(LTNode* phead)
//{
//	assert(!LTEmpty(phead));
//
//	LTEmpty(phead->next);
//}

//尾删
void LTPopBack(LTNode* phead)
{
	assert(!LTEmpty(phead));

	LTNode* tail = phead->prev;
	LTNode* tailPrev = tail->prev;

	tailPrev->next = phead;
	phead->prev = tailPrev;
	free(tail);
}

尾删
//void LTPopBack(LTNode* phead)
//{
//	assert(!LTEmpty(phead));
//
//	LTEmpty(phead->prev);
//}

//查找
LTNode* LTFind(LTNode* phead, ListNodeDataType x)
{
	assert(phead);

	LTNode* cur = phead->next;;
	while (cur != phead)
	{
		if (cur->data == x)
		{
			return cur;
		}
		
		cur = cur->next;
	}

	return NULL;
}

//修改
void LTModify(LTNode* phead, LTNode* pos, ListNodeDataType x)
{
	assert(phead);
	assert(pos);

	pos->data = x;
}

//在pos之前插入
void LTInsert(LTNode* pos, ListNodeDataType x)
{
	assert(pos);

	LTNode* prev = pos->prev;
	LTNode* newnode = BuyLTNode(x);
	
	prev->next = newnode;
	newnode->prev = prev;
	
	newnode->next = pos;
	pos->prev = newnode;
}

//删除pos位置的值
void LTErase(LTNode* pos)
{
	assert(pos);

	LTNode* posPrev = pos->prev;
	LTNode* posNext = pos->next;

	posPrev->next = posNext;
	posNext->prev = posPrev;
	free(pos);
}

🥝Test.c

#include "List.h"

//头插测试
void TestList01()
{
	LTNode* plist = LTInit();

	LTPushFront(plist, 4);
	LTPushFront(plist, 3);
	LTPushFront(plist, 2);
	LTPushFront(plist, 1);

	LTPrint(plist);

	LTDestroy(plist);
	plist = NULL;
}

//尾插测试
void TestList02()
{
	LTNode* plist = LTInit();

	LTPushBack(plist, 4);
	LTPushBack(plist, 3);
	LTPushBack(plist, 2);
	LTPushBack(plist, 1);

	LTPrint(plist);

	LTDestroy(plist);
	plist = NULL;
}

//头删测试
void TestList03()
{
	LTNode* plist = LTInit();

	LTPushFront(plist, 4);
	LTPushFront(plist, 3);
	LTPushFront(plist, 2);
	LTPushFront(plist, 1);
	LTPopFront(plist);

	LTPrint(plist);

	LTDestroy(plist);
	plist = NULL;
}

//尾删测试
void TestList04()
{
	LTNode* plist = LTInit();

	LTPushFront(plist, 4);
	LTPushFront(plist, 3);
	LTPushFront(plist, 2);
	LTPushFront(plist, 1);
	LTPopBack(plist);

	LTPrint(plist);

	LTDestroy(plist);
	plist = NULL;
}

//查找修改测试
void TestList05()
{
	LTNode* plist = LTInit();

	LTPushFront(plist, 4);
	LTPushFront(plist, 3);
	LTPushFront(plist, 2);
	LTPushFront(plist, 1);
	
	LTNode* pos = LTFind(plist, 3);
	if (pos)
	{
		LTModify(plist, pos, 8);
	}

	LTPrint(plist);

	LTDestroy(plist);
	plist = NULL;
}

//在pos之前插入
void TestList06()
{
	LTNode* plist = LTInit();

	LTPushFront(plist, 4);
	LTPushFront(plist, 3);
	LTPushFront(plist, 2);
	LTPushFront(plist, 1);

	LTNode* pos = LTFind(plist, 3);
	if (pos)
	{
		LTInsert(pos, 7);
	}

	LTPrint(plist);

	LTDestroy(plist);
	plist = NULL;
}

//删除pos位置的值
void TestList07()
{
	LTNode* plist = LTInit();

	LTPushFront(plist, 4);
	LTPushFront(plist, 3);
	LTPushFront(plist, 2);
	LTPushFront(plist, 1);

	LTNode* pos = LTFind(plist, 2);
	if (pos)
	{
		LTErase(pos);
	}

	LTPrint(plist);

	LTDestroy(plist);
	plist = NULL;
}



int main()
{
	//TestList01();
	//TestList02();
	//TestList03();
	//TestList04();
	//TestList05();
	//TestList06();
	//TestList07();

	return 0;
}

🥰这期内容比较容易一些而且比较有趣,希望烙铁们可以理解消化哦!

总结🥰
以上就是 【数据结构】带头双向循环链表—C语言版 的全部内容啦🥳🥳🥳🥳
本文章所在【数据结构与算法】专栏,感兴趣的烙铁可以订阅本专栏哦🥳🥳🥳
前途很远,也很暗,但是不要怕,不怕的人面前才有路。💕💕💕
小的会继续学习,继续努力带来更好的作品😊😊😊
创作写文不易,还多请各位大佬uu们多多支持哦🥰🥰🥰

请添加图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

只考一门数据结构,计算机学硕复录比1:1的山东双非学校考情分析

青岛理工大学 考研难度&#xff08;☆&#xff09; 内容&#xff1a;23考情概况&#xff08;拟录取和复试分析&#xff09;、院校概况、23专业目录、23复试详情、各专业考情分析、各科目考情分析。 正文1420字&#xff0c;预计阅读&#xff1a;3分钟 2023考情概况 青岛理工…

CI/CD 持续集成 持续交付

CI&#xff08;Continuous integration&#xff09;持续集成 参考&#xff1a;https://www.jianshu.com/p/2132949ff84a 持续集成是指多名开发者在开发不同功能代码的过程当中&#xff0c;可以频繁的将代码行合并到一起并切相互不影响工作。 持续集成的目的&#xff0c;是让…

ATF(TF-A)安全通告 TFV-2 (CVE-2017-7564)

安全之安全(security)博客目录导读 ATF(TF-A)安全通告汇总 目录 一、ATF(TF-A)安全通告 TFV-2 (CVE-2017-7564) 二、 CVE-2017-7564 一、ATF(TF-A)安全通告 TFV-2 (CVE-2017-7564) Title 启用安全自托管侵入式调试接口&#xff0c;可允许非安全世界引发安全世界panic CV…

SpringCloud学习笔记(十二)_Zipkin全链路监控

Zipkin是SpringCloud官方推荐的一款分布式链路监控的组件&#xff0c;使用它我们可以得知每一个请求所经过的节点以及耗时等信息&#xff0c;并且它对代码无任何侵入&#xff0c;我们先来看一下Zipkin给我们提供的UI界面都是提供了哪些信息。 如何使用Zipkin 虽然在SpringBoot…

C语言练习题解析:挑战与突破,开启编程新篇章!(2)

&#x1f493;博客主页&#xff1a;江池俊的博客⏩收录专栏&#xff1a;C语言刷题专栏&#x1f449;专栏推荐&#xff1a;✅C语言初阶之路 ✅C语言进阶之路&#x1f4bb;代码仓库&#xff1a;江池俊的代码仓库&#x1f389;欢迎大家点赞&#x1f44d;评论&#x1f4dd;收藏⭐ 文…

phpstorm动态调试

首先在phpstudy搭建好网站&#xff0c;在管理拓展开启xdebug拓展 查看php.ini配置已经更改 需要增添修改一下设置 [Xdebug] zend_extensionD:/phpstudy_pro/Extensions/php/php5.6.9nts/ext/php_xdebug.dll xdebug.collect_params1 xdebug.collect_return1 xdebug.auto_trace…

【大数据】Doris:基于 MPP 架构的高性能实时分析型数据库

Doris&#xff1a;基于 MPP 架构的高性能实时分析型数据库 1.Doris 介绍 Apache Doris 是一个基于 MPP&#xff08;Massively Parallel Processing&#xff0c;大规模并行处理&#xff09;架构的高性能、实时的分析型数据库&#xff0c;以极速易用的特点被人们所熟知&#xff…

IP协议分片重组问题

分片是什么&&为什么会有分片 IP数据报分片的主要目的是为了防止IP数据报文长度超过下一跳链路MTU(最大传输单元)。 数据链路层之MTU 数据链路层中有一个东西叫做MTU&#xff08;最大传输单元&#xff09;&#xff0c;它的作用主要是控制上层给的数据报不要太大&#…

C语言网络编程:实现自己的高性能网络框架

一般生产环境中最耗时的其实是业务逻辑处理。所以&#xff0c;是不是可以将处理业务逻辑的代码给拆出来丢到线程池中去执行。 比如像下面这样&#xff1a; ​我们事先创建好一堆worker线程&#xff0c;主线程accepter拿到一个连接上来的套接字&#xff0c;就从线程池中取出一个…

Docker harbor 私有仓库的部署和管理

目录 一、什么是Harbor 二、Harbor的特性 三、Harbor的构成 四、部署配置Docker Harbor 1. 首先需要安装 Docker-Compose 服务 2.部署 Harbor 服务 3.使用harbor仓库 &#xff08;1&#xff09;项目管理 &#xff08;2&#xff09;用户管理 一、什么是Harbor Harbor …

2023高教社杯数学建模思路 - 复盘:校园消费行为分析

文章目录 0 赛题思路1 赛题背景2 分析目标3 数据说明4 数据预处理5 数据分析5.1 食堂就餐行为分析5.2 学生消费行为分析 建模资料 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 赛题背景 校园一卡通是集…

使用rem + sass + 媒体查询 进行横竖屏适配移动端项目

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、 媒体查询&#xff08;Media Query&#xff09;1.简单了解2.简单例子3. 语法&#xff08;1&#xff09;媒体类型&#xff08;mediatype &#xff09;&#xff0…

【USRP】调制解调系列3:2FSK、4FSK、8FSK,基于labview的实现

2FSK、4FSK、8FSK FSK&#xff08;Frequency-shift keying&#xff09;是信息传输中使用得较早的一种调制方式,它的主要优点是: 实现起来较容易,抗噪声与抗衰减的性能较好。在中低速数据传输中得到了广泛的应用。最常见的是用两个频率承载二进制1和0的双频FSK系统。 FSK 信号…

基于JAVA SpringBoot互联网就医门诊挂号管理系统

摘要 随着时代的发展,无线互联网技术的应用和普及给人们的生活带来了极大的改变,现在信息技术不仅可以提高我们的工作效率,还能有效的规避一些错误风险,节约人力成本。我国国民一方面对健康的要求越来越重视了&#xff0c;另一方面现代人的健康问题日益严重&#xff0c;所以医院…

图神经网络和分子表征:3. 不变网络最后的辉煌

写这篇文章的时候已经是2023年的8月份&#xff0c;GNN for molecule property prediction 这个小领域正在变得火热起来&#xff0c;各大榜单被不断刷新&#xff0c;颇有当年 CNN 刷榜 imagenet 的势头。 由于对力、维里等性质有着内禀优势&#xff0c;当下高居榜首的模型毫无疑…

Nginx从入门到精通(超级详细)

文章目录 一、什么是Nginx1、正向代理2、反向代理3、负载均衡4、动静分离 二、centos7环境安装Nginx1、安装依赖2、下载安装包3、安装4、启动5、停止 三、Nginx核心基础知识1、nginx核心目录2、常用命令3、默认配置文件讲解4、Nginx虚拟主机-搭建前端静态服务器5、使用nignx搭建…

解决OpenCV的GStreamer warning警告

调用 cv::VideoCapture出现的警告&#xff1a; [ WARN:0] global ../modules/videoio/src/cap_gstreamer.cpp (1758) handleMessage OpenCV | GStreamer warning: Embedded video playback halted; module v4l2src0 reported: Internal data stream error. [ WARN:0] global .…

TCP协议基础

一&#xff1a; TCP协议是什么&#xff1f; TCP协议是基于面向连接&#xff0c;可靠传输&#xff0c;基于字节流的传输层通信协议 1. 面向连接 TCP协议是一种面向连接的协议&#xff0c;意味着在双方在建立数据传输之前&#xff0c;需要进行一个逻辑上的连接&#xff0c;且是…

智慧矿山2.0:煤矿智能化综合管理AI大数据监管平台建设方案设计

一、行业背景 能源与煤矿是我国国民经济的重要物质生产部门和支柱产业之一&#xff0c;同时也是一个安全事故多发的高危行业&#xff0c;施工阶段的现场管理对工程成本、进度、质量及安全等至关重要。煤矿智能化既是未来趋势&#xff0c;更是产业发展需求&#xff0c;建设智慧…

C语言——程序执行的三大流程

顺序 : 从上向下&#xff0c; 顺序执行代码分支 : 根据条件判断&#xff0c; 决定执行代码的分支循环 : 让特定代码重复的执行