【数据结构和算法初阶(C语言)】队列实操(概念实现+oj题目栈和队列的双向实现以及循环链表难点题目详解!)

news2025/1/6 19:43:15

 

目录

1. 队列的概念及结构

2.队列结构存在的意义应用

3.队列实现的结构选择

4.队列实现

5.队列对数据的处理

5.1队列初始化

5.2队尾入数据

5.3队头出数据

5.4获取队列尾部元素

5.5获取队列头部元素

5.6获取队列中元素个数

5.7检测队列是否为空

5.8销毁队列

6.循环队列补充

7.使用队列实现栈

8.使用栈实现队列

9.实现循环队列 

​编辑

10.结语


1. 队列的概念及结构

队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,

队列具有先进先出 FIFO(First In First Out)

入队列:进行插入操作的一端称为队尾

出队列:进行删除操作的一端称为队头

2.队列结构存在的意义应用

①公平排队(决定公平性的东西)

不会出现插队问题(不过有竞争问题,两个窗口同时叫或者两个号码一起出---操作系统加速解决)

比如医院或者银行的排号---抽号机

②BFS广度优先

树,迷宫实现等

3.队列实现的结构选择

数组和链表都可以实现队列,但是链表的头插尾插,头删尾删要方便些,所以首选链表

 单向还是双向:选择单向,双向的优势是方便找前一个1节点,没有这个需求。(找尾不是因为双向,双向循环方便找尾,但是单向加一个尾巴指针也可以解决)

是否需要带哨兵位的链表:哨兵位是为了解决二级指针(可以将头尾指针封装为结构体进行传参,这样就可以改变真实的指针了,所以哨兵位可要可不要),尾插少一次判断。

选择单向不循环链表即可实现。

4.队列实现

typedef  int QdataType;


typedef struct QListNode
{
	struct QListNode* next;
	QdataType data;
}QNode;

//将头尾指针封装为一个结构体,解决传递二级指针的问题

typedef struct Queue
{
	QNode* head;
	QNode* tail;
	int size;//保存链表的大小
}Queue;

5.队列对数据的处理

5.1队列初始化

void QUeueInit(Queue* pq)//初始化队列
{
	assert(pq);
	pq->head = pq->tail = NULL;
	pq->size = 0;
}

5.2队尾入数据

void QueuePush(Queue* pq, QdataType x)//队列增加数据
{
	assert(pq);//首先传入的这个结构体要存在
	//申请到节点来创建
	QNode* newnode = (QNode*)malloc(sizeof(QNode));
	if (newnode == NULL)
	{
		perror("malloc");
	}
	//新节点初始化
	newnode->data = x;
	newnode->next = NULL;
	//准备插入,看一下是不是第一次插入
	if (pq->tail == NULL)
	{
		pq->head = pq->tail = newnode;
		pq->size++;
	}
	else
	{
		pq->tail->next = newnode;
		pq->tail = pq->tail->next;
		pq->size++;
	}
}

5.3队头出数据

void QueuepPop(Queue* pq, QdataType x)//队列删除元素
{
	assert(pq);
	assert(pq->size != 0);
	if (pq->head->next == NULL)//因为不是带哨兵位的,删除到最后一个位置防止尾指针成为野指针,单独处理
	{
		free(pq->head);
		pq->head = pq->tail = NULL;
		pq->size--;
	}
	else
	{
		QNode* next = pq->head->next;
		free(pq->head);
		pq->head = next;
		pq->size--;
	}
	
}

5.4获取队列尾部元素

QdataType QueueBack(Queue* pq)//获取队列前后面元素
{
	assert(pq);
	assert(!QueueEmpty);
	return pq->tail->data;
}

5.5获取队列头部元素

QdataType QueueFront(Queue* pq)//获取队列前面元素
{
	assert(pq);
	assert(!QueueEmpty);
	return pq->head->data;
}

5.6获取队列中元素个数

int QueueSize(Queue* pq)
{
	assert(pq);
	return pq->size;
}

5.7检测队列是否为空

bool QueueEmpty(Queue* pq)//检测队列是否为空
{
	assert(pq);
	return pq->head == NULL;
}

5.8销毁队列

void QueueDestory(Queue* pq)//销毁队列
{
	assert(pq);
	QNode* cur = pq->head;
	while (cur)
	{
		QNode* next = cur->next;
		free(cur);
		cur = cur->next;
	}
	pq->head = pq->tail = NULL;
	pq->size = 0;
}

6.循环队列补充

设计你的循环队列实现。 循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环。它也被称为“环形缓冲器”。

循环队列的一个好处是我们可以利用这个队列之前用过的空间。在一个普通队列里,一旦一个队列满了,我们就不能插入下一个元素,即使在队列前面仍有空间。但是使用循环队列,我们能使用这些空间去存储新的值。

7.使用队列实现栈

链接跳转题目:

. - 力扣(LeetCode). - 备战技术面试?力扣提供海量技术面试资源,帮助你高效提升编程技能,轻松拿下世界 IT 名企 Dream Offer。https://leetcode.cn/problems/implement-stack-using-queues/

分析:栈的结构特点是数据先进后出

           队列的结构特点是先进先出

那么就是说我们需要使用先进先出的结构来实现后进先出的结构:

这是初始状态:

对于队列来说先出1,对于栈来说先出4,我们现在就是要想办法利用队列的函数实现我们的先出4:

我们先出123将其放进队列2,然后单独删除4是不是就实现了后进先出。

那么后续有新数据要进入我们自己的“栈”直接进入有元素的队列尾进就好了,然后一样的办法进行出栈。如果两个队列都没有元素,数据放入那个队列都可以

​ 出栈:

过程清楚我们来上代码:主逻辑代码:

结构图:

8.使用栈实现队列

做题地址

原理:一个栈作为数据进入,另外一个栈作为数据流出

. - 力扣(LeetCode)

队列是先进先出,栈是先进后出:

这是初始状态

如果是队列1那么出数据的顺序就是1.2.3.4

栈的出数据顺序是4.3.2.1

我们的思路是:将数据出到栈2,然后从栈顶出数据就可以实现一次数据的顺序改变

当后续插入数据的时候:我们应该往空的栈里面去插入数据,一直到我们的这个非空栈数据出完了在进行数据移值:

实现原理明白了;动手写代码实现

结构图:

9.实现循环队列 

首先分析链表实现这个循环列表的难易程度:

首先考虑单向链表,带头尾指针:

链表是可以实行的,不过一开始就要创建一个循环链表还是有些麻烦,

介绍巧解决方案:使用数组多开一个空间法:

 

数组麻烦的地方就是回绕的时候要多判断一次。

用数组实现,

判满和判空

插入数据前要先判断满没满

特别注意就是尾指针在最后的情况,统一模当然也可以担当rear = K+1的时候,直接置0.

删除数据前要判断空不空 

取出头数据,先判空

取尾数据

通过;附上源代码

typedef struct {
    int * a ;//队列的空间
    int k;//队列长度
    int front;//头的下标
    int rear;//尾的下标

    
} MyCircularQueue;


MyCircularQueue* myCircularQueueCreate(int k) {
    //为结构体1申请空间
    MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));
    //为队列申请空间,多申请一个,如果是链表这里就要循环申请node,在串起来
    obj->a = (int*)malloc(sizeof(int)*(k+1));
    //然后我们的头尾下标目前没有元素放在最开始
    obj->front = obj->rear = 0;
    obj->k = k;
    return obj;

    
}
bool myCircularQueueIsEmpty(MyCircularQueue* obj) {
    //我们的rear尾指针和头指针相等的时候就满
    return obj->front == obj->rear;
    
}
bool myCircularQueueIsFull(MyCircularQueue* obj) {
    //当尾巴指针的下一个,这里由于是数组,是通过下标确定,所以是下标之间的关系要重叠起来
    return (obj->rear+1)%(obj->k+1)==obj->front;
}

bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {
    //首先就要判断满没满
    if(myCircularQueueIsFull(obj))
    {
        return false;
    }
    obj->a[obj->rear] = value;
   //obj->rear++;要解决尾部指针来到最后一个位置的时候要回到最开始
   obj->rear++;//坐标先加加
   //模一下
   obj->rear %= (obj->k+1);
   return true;
}




bool myCircularQueueDeQueue(MyCircularQueue* obj) {
    //首先要判断空不空

    if(myCircularQueueIsEmpty(obj))
    {
        return false;
    }
    //让后删除一个,是从头删除,那么头指针就要往后走一下,然后也要注意绕回来的问题
    //不用抹除数据,之间角标加加,到时候插入数据,值会覆盖掉
    obj->front++;
    obj->front %= (obj->k+1);
    return true;
}

int myCircularQueueFront(MyCircularQueue* obj) {
    //先判断空
    if(myCircularQueueIsEmpty(obj))
    return -1;
    else
    {
        return obj->a[obj->front];
    }
    
}

int myCircularQueueRear(MyCircularQueue* obj) {
    //正常队尾就是rear-1
    //但是rear在开头,-1就是-1了,可以当rear=0,直接变k+1
    //也可以:(rear+k)%(k+1)  
    if(myCircularQueueIsEmpty(obj))
    {
        return -1;
    }
    else{
        return obj->a[(obj->rear+obj->k)%(obj->k+1)];
    }
    
}



void myCircularQueueFree(MyCircularQueue* obj) {
    free(obj->a);
    free(obj);
    
}

/**
 * Your MyCircularQueue struct will be instantiated and called as such:
 * MyCircularQueue* obj = myCircularQueueCreate(k);
 * bool param_1 = myCircularQueueEnQueue(obj, value);
 
 * bool param_2 = myCircularQueueDeQueue(obj);
 
 * int param_3 = myCircularQueueFront(obj);
 
 * int param_4 = myCircularQueueRear(obj);
 
 * bool param_5 = myCircularQueueIsEmpty(obj);
 
 * bool param_6 = myCircularQueueIsFull(obj);
 
 * myCircularQueueFree(obj);
*/

 

10.结语

以上就是本次分享的所有内容,三个题目都有一定的难度对栈和队列的知识考察得综合,一定要理清二者的概念。

创作不易,大家如果觉得还可以的话,欢迎大家三连,有问题的地方欢迎大家指正,一起交流学习,一起成长,我是Nicn,正在c++方向前行的奋斗者,数据结构内容持续更新中,感谢大家的关注与喜欢。

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

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

相关文章

C语言之快速排序

目录 一 简介 二 代码实现 快速排序基本原理: C语言实现快速排序的核心函数: 三 时空复杂度 A.时间复杂度 B.空间复杂度 C.总结: 一 简介 快速排序是一种高效的、基于分治策略的比较排序算法,由英国计算机科学家C.A.R. H…

矩阵中移动的最大次数

文章目录 所属专栏:BFS算法 题目链接 思路如下: 1.首先我们需要从第一列开始遍历,寻找每一个都能够满足条件的位置,将它插入到数组里面 2.第一列遍历完了后我们先判断第一列的数是否都满足条件插入到数组里面,如果数组为空&#…

03.事件进阶

一、事件流 事件流是对事件执行过程的描述,了解事件的执行过程有助于加深对事件的理解,提升开发实践中对事件运用的灵活度。 如上图所示,任意事件被触发时总会经历两个阶段:【捕获阶段】和【冒泡阶段】。 简言之,捕获…

肝了三天,完成了AIGC工具网站大全,建议收藏再看

说是肝了三天,其实远远不止,前前后后,从资料搜集到最后整理成文,有近一个月了,大家看在整理不易的份上,给点个赞吧,不要光顾着收藏呀! 国内网站 AIGC 导航 https://www.aigc.cn 网…

wayland(xdg_wm_base) + egl + opengles 渲染使用纹理贴图的旋转 3D 立方体实例(十三)

文章目录 前言一、使用 stb_image 库加载纹理图片1. 获取 stb_image.h 头文件2. 使用 stb_image.h 中的相关接口加载纹理图片3. 纹理图片——cordeBouee4.jpg二、渲染使用纹理贴图的旋转 3D 立方体1. egl_wayland_texture_cube.c2. Matrix.h 和 Matrix.c3. xdg-shell-client-pr…

使用 pnpm 搭建 monorepo 项目

引言 在我之前的开发经历中,并没有实际使用过 Monorepo 管理项目,尽管之前对此有所了解,但并未深入探究。然而,如今许多开源项目都采纳了 Monorepo 方式,对于不熟悉它的开发者来说,阅读和理解这些项目的源…

关于使用过别人的GitHub/Gitee账号提交东西到仓库后,自己的仓库提交失败问题的解决方法

目录 引言 问题出现的原因 问题的解决方法 打开控制面板 方法一: 方法二: 进入Windows凭据管理器 结语 引言 这个问题折磨了我将近三个小时,为了减少有人和我一样走弯路且防止自己忘记解决方法,便有了这篇文章。 问题出现…

【洛谷 P8602】[蓝桥杯 2013 省 A] 大臣的旅费 题解(图论+深度优先搜索+树的直径+链式前向星)

[蓝桥杯 2013 省 A] 大臣的旅费 题目描述 很久以前,T 王国空前繁荣。为了更好地管理国家,王国修建了大量的快速路,用于连接首都和王国内的各大城市。 为节省经费,T 国的大臣们经过思考,制定了一套优秀的修建方案&am…

uv 必备的工具 ps ai 全家桶合集

非常稀有的资源 ,必应搜索 易品资源yipinziyuan 可以找到

浅谈如何自我实现一个消息队列服务器(1)——需求分析

文章目录 一、什么是消息队列?二、当下主流的消息队列(MQ)三、自我实现一个消息队列服务器的前期准备——需求分析3.1 核心概念3.2 broker server 核心概念3.2.1、虚拟主机(Virtual Host)3.2.2、交换机(Exchange)3.2.2…

如何构建Docker自定义镜像

说明:平常我们使用Docker运行各种容器,极大地方便了我们对开发应用的使用,如MySQL、Redis,以及各种中间件,使用时只要拉镜像,运行容器即可。本文介绍如何创建一个Demo,自定义构建一个镜像。 开…

基于 K8s 容器集群的容灾架构与方案

作者:庄宇 在设计系统架构时,我们必须假设任何组件和任何基础设施可能会在任何时间失效,例如:自然灾害,电力中断,网络中断,错误的系统变更等。为了应对挑战,我们必须设计合适的容灾…

HTML基础:head头部标签包含的8种形式详解

你好&#xff0c;我是云桃桃。 HTML <head> 标签用于定义文档的头部&#xff0c;包含了一些元数据和引用的外部资源&#xff0c;但并不会直接展示在页面上。<head> 标签位于 <html> 标签内部&#xff0c;紧跟在 <html> 开始标签之后&#xff0c;在 &l…

贪心算法(两个实例)

例一&#xff1a;调度问题 问题&#xff1a;由n项任务&#xff0c;每项任务的加工时间已知&#xff0c;从零时刻开始陆续加入一台机器上去加工&#xff0c;每个任务完成的时间是从0时刻到任务加工截至的时间。 求总完成时间&#xff08;所有任务完成时间最短计划方案&#xf…

C语言字符函数和字符串函数详解

Hello, 大家好&#xff0c;我是一代&#xff0c;今天给大家带来有关字符函数和字符串函数的有关知识 所属专栏&#xff1a;C语言 创作不易&#xff0c;望得到各位佬们的互三呦 一.字符函数 在C语言中有一些函数是专门为字符设计的&#xff0c;这些函数的使用都需要包含一个头文…

SQLiteC/C++接口详细介绍之sqlite3类(十七)

返回目录&#xff1a;SQLite—免费开源数据库系列文章目录 上一篇&#xff1a;SQLiteC/C接口详细介绍之sqlite3类&#xff08;十六&#xff09; 下一篇&#xff1a; SQLiteC/C接口详细介绍之sqlite3类&#xff08;十八&#xff09; ​ 53.sqlite3_trace_v2 函数功能&#x…

EXCEL+PYTHON学习3

1&#xff09; 遍历一个SHEET&#xff0c;无非就是两个循环&#xff0c;rows属性是取得所有行。 fn data3_16.xlsx wb openpyxl.load_workbook(fn) ws wb.active for row in ws.rows:for cell in row:print(cell.value, end )print() 2&#xff09; 返回工作表的最小行数…

【Leetcode-54.螺旋矩阵】

题目&#xff1a; 给你一个 m 行 n 列的矩阵 matrix &#xff0c;请按照 顺时针螺旋顺序 &#xff0c;返回矩阵中的所有元素。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,2,3],[4,5,6],[7,8,9]] 输出&#xff1a;[1,2,3,6,9,8,7,4,5]示例 2&#xff1a; 输入&#xff1…

Day38:安全开发-JavaEE应用SpringBoot框架MyBatis注入Thymeleaf模版注入

目录 SpringBoot-Web应用-路由响应 SpringBoot-数据库应用-Mybatis SpringBoot-模版引擎-Thymeleaf 思维导图 Java知识点 功能&#xff1a;数据库操作&#xff0c;文件操作&#xff0c;序列化数据&#xff0c;身份验证&#xff0c;框架开发&#xff0c;第三方库使用等. 框架…

基于springboot实现房源出租信息系统项目【项目源码+论文说明】

基于springboot实现房源出租信息系统演示 摘要 近些年来在一线城市的房子需求量在逐步递增&#xff0c;其中租房子和出租房子的需求业务不断增加。那么租房对于我们一线二线城市来说是一个非常大&#xff0c;而且具有经济提升能力的业务场景。那么信息技术在此行业的加持早已是…