【脚踢数据结构】队列(顺序和链式)

  • (꒪ꇴ꒪ ),Hello我是祐言QAQ
  • 我的博客主页:C/C++语言,Linux基础,ARM开发板,软件配置等领域博主🌍
  • 快上🚘,一起学习,让我们成为一个强大的攻城狮!
  • 送给自己和读者的一句鸡汤🤔:集中起来的意志可以击穿顽石!
  • 作者水平很有限,如果发现错误,可在评论区指正,感谢🙏


        在我们的日常生活中,队列是一个非常常见的现象。无论是在商店结账,还是在公交站等车,我们都在使用队列。在计算机科学中,队列也是一个重要的数据结构,用于处理和组织数据。在这篇文章中,我们将详细探讨队列的定义、操作、以及如何用C语言实现队列。

一、队列的定义


        队列(Queue)是一种特殊类型的线性数据结构,它遵循特定的操作规则,即遵循“先进先出”(FIFO,First-In-First-Out)原则。这意味着在队列中,首先加入的元素将会首先被移除,最后加入的元素将会最后被移除。

         当我们想存入1时,先移动front(队头)然后再写入数据1,拿数据也是一样,但务必保证先移动rear(队尾),再拿出数据,否则将会错位导致出错。

二、顺序队列

      1.  队列结构体定义


        首先需要定义一个顺序队列,我们可以使用结构体来定义一个队列,它包含一个数组(用于存储队列的数据)和三个整数(用于表示队列长度、队首和队尾的位置)。


typedef int Datatype;

//队列的结构体定义
typedef struct Quene
{
	Datatype *q;	//用来存放队列的数据
	int size;		//队列的长度
	int front;		//队头
	int rear;		//队尾
}queue;

    2.  初始化队列


        接下来,我们需要初始化队列。在初始化时,我们将队首和队尾都设置为0,表示队列为空。

//初始化一个队列
queue *init_queue(int size)
{
	queue *que = malloc(sizeof(queue));
	if (que!=NULL)
	{
		que->q = calloc(size, sizeof(Datatype));
		que->size = size;
		que->front = 0;
		que->rear = 0;
	}
	return que;
}

    3.  队空和队满


        如果我们想要实现入队和出队操作,我们需要先考虑队列可能会溢出或下溢的情况,因此我们需要判断是否队空或队满。

//队空判断
bool isempty_queue(queue *q)
{
	return (q->rear == q->front);
}

//队满判断
bool isfull_queue(queue *q)
{
	return ((q->rear+1)%q->size == q->front);
}

    4.  入队和出队

//入队
bool en_queue(queue *que, Datatype data)
{
	if (isfull_queue(que))
	{
		return false;
	}
	//先挪rear
	que->rear = (que->rear+1)%(que->size);
	//再入数据
	que->q[que->rear] = data;
	return true;
}

//出队
bool de_queue(queue *que, Datatype *data)
{
	if (isempty_queue(que))
	{
		return false;
	}
	//先挪front
	que->front = (que->front+1)%(que->size);
	//再取数据
	*data = que->q[que->front];
	return true;
}


        队列是计算机科学中的一个基础概念,它在许多场景中都有应用,如操作系统的任务调度,网络的数据包处理等。理解队列的工作原理并能够实现队列,对于学习和理解计算机科学的其他概念是非常有帮助的。希望这篇文章能帮助你理解和实现队列。

        下面是一个简单的顺序队列举例,实现:输入正整数,添加员工信息,入队,用这个正整数表示员工号;输入负整数,出队(队首),显示该员工的所有信息;否则就退出。

        员工信息:工号、姓名、工资

        完整源码:

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

#define SIZE 1024
typedef struct people
{
	int number;		//工号
	char name[20];	//姓名
	int money;		//工资
}Datatype;

//队列的结构体定义
typedef struct Quene
{
	Datatype *q;	//用来存放队列的数据
	int size;		//队列的长度
	int front;		//队头
	int rear;		//队尾
}queue;

//初始化一个队列
queue *init_queue(int size)
{
	queue *que = malloc(sizeof(queue));
	if (que!=NULL)
	{
		que->q = calloc(size, sizeof(Datatype));
		que->size = size;
		que->front = 0;
		que->rear = 0;
	}
	return que;
}

//队空判断
bool isempty_queue(queue *q)
{
	return (q->rear == q->front);
}

//队满判断
bool isfull_queue(queue *q)
{
	return ((q->rear+1)%q->size == q->front);
}

//入队
bool en_queue(queue *que, Datatype data)
{
	if (isfull_queue(que))
	{
		return false;
	}
	//先挪rear
	que->rear = (que->rear+1)%(que->size);
	//再入数据
	que->q[que->rear] = data;
	return true;
}

//出队
bool de_queue(queue *que, Datatype *data)
{
	if (isempty_queue(que))
	{
		return false;
	}
	//先挪front
	que->front = (que->front+1)%(que->size);
	//再取数据
	*data = que->q[que->front];
	return true;
}

// 添加信息
void init_info(Datatype *data,int n)
{
	data->number = n;
	printf("请输入工人信息\n");
	while(getchar() != '\n');
	printf("姓名:");
	scanf("%s", data->name);
	printf("工资:");
	scanf("%d", &data->money);
}

int main(int argc, char const *argv[]) {

    queue *q = init_queue(SIZE);
    int n;
    Datatype data;
    

    while (1) 
	{
		printf("请输入一个正整数或负数\n");
		scanf("%d", &n);
		
		if (n > 0){
			init_info(&data, n);
			en_queue(q, data);
			continue;
		}
		else if(n < 0){
			Datatype d;
			if (de_queue(q, &d)) {
				printf("工号:%d,姓名:%s,工资:%d\n", d.number, d.name, d.money);
			} else {
				printf("队列已空,无法出队。\n");
			}
		}
		else{
			return -1;
		}
	}
    free(q->q);
    free(q);

    return 0;
}

三、链式队列

        链式队列(Linked Queue)是一种使用链表来实现的队列数据结构。与顺序队列不同,链式队列的元素并不直接存储在数组中,而是通过链表节点来连接。

        并且 由于使用链表实现,链式队列的大小可以根据需要动态分配和释放内存,避免了固定数组大小可能带来的限制。因此就没有是否队满问题

1.结构体定义

typedef int Datatype;

typedef struct Node
{
	Datatype data;
	struct Node *next;
}node;

typedef struct List_queue
{
	node *rear;		//队尾指针
	node *front;	//队头指针
	int size;		//链式队列的长度(实际的元素的个数)
}L_q;

2.创建新节点和判断队空

//创建新节点
node *create_node(Datatype data)
{
	node *new = malloc(sizeof(node));
	if (new != NULL)
	{
		new->data = data;
		new->next = NULL;
	}
	return new;
}
//链式队列是否为空
bool isempty_list_queue(L_q *q)
{
	return (q->size == 0);
}

3.初始化队列

//初始化链式队列
L_q *init_list_queue()
{
	L_q *q = malloc(sizeof(L_q));
	if (q!=NULL)
	{
		q->rear = NULL;
		q->front = NULL;
		q->size = 0;
	}
	return q;
}

3.入队

        入队操作在链表的末尾添加一个新节点,同时更新队尾指针。

//入队
bool en_list_queue(L_q *q, Datatype data)
{
	//先要将数据创建新节点
	node *new = create_node(data);
	if (new==NULL)
	{
		return false;
	}
	//如果是第一次入队,new节点既是队尾,也是队头
	if (isempty_list_queue(q))
	{
		q->rear = new;
		q->front = new;
	}
	else    //不是第一次入队
	{
		//先将尾部节点的next指向new节点
		q->rear->next = new;
		//尾部节点要变成新节点new
		q->rear = new;
	}
	//队的元素个数要+1
	q->size++;
	return true;
}

4.出队

         出队操作移除链表的第一个节点,同时更新队头指针。

//出队
bool de_list_queue(L_q *q, Datatype *data)
{
	if (isempty_list_queue(q))
	{
		return false;
	}
	else if(q->size == 1)//只有一个数据的时候
	{
		q->rear = NULL;
	}
	//在链表不为空的情况下,先拿队头的数据
	*data = q->front->data;
	//将队头指向下一个节点
	q->front = q->front->next;
	//链式队列的元素个数-1
	q->size--;
	return true;
}

 5.遍历

//遍历
void display(L_q *q)
{
	if (q->front == NULL)
	{
		return ;
	}
	node *p = q->front;
	while(p->next != NULL)
	{
		printf("%d ", p->data);
		p = p->next;
	}
	printf("%d\n", p->data);
}

        简单示例:当我们输入正数时,入队并遍历整个队列;当我们输入负数时,出队一个元素,并再次遍历队列;输入0时退出。

int main(int argc, char const *argv[])
{
	L_q *q = init_list_queue();
	int num;
	Datatype data;
	while(1)
	{
		scanf("%d", &num);
		if(num > 0)
		{
			en_list_queue(q, num);	
			display(q);	
		}
		else if(num < 0)
		{
			de_list_queue(q, &data);
			display(q);	
		}
		else
		{
			break;
		}
	}
	return 0;
}

四、结语

        
        队列作为一种基本的数据结构,在我们的编程生涯中扮演着重要的角色。希望这篇文章提供了一个清晰、详细的队列概述,帮助你理解队列的基本概念和操作,以及如何用C语言实现队列。

        选择顺序队列还是链式队列取决于实际应用的需求。如果你需要一个固定大小的队列,可以考虑使用顺序队列。如果你希望队列大小能够根据需要进行动态调整,那么链式队列更适合。在大多数情况下,链式队列具有更好的扩展性和灵活性。

        更多C语言Linux系统ARM板实战数据结构相关文章,关注专栏:

   手撕C语言

            玩转linux

                    脚踢数据结构

                            6818(ARM)开发板实战

📢写在最后

  • 今天的分享就到这啦~
  • 觉得博主写的还不错的烦劳 一键三连喔~
  • 🎉感谢关注🎉

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

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

相关文章

接口测试工具——Postman测试工具 Swagger接口测试+SpringBoot整合 JMeter高并发测试工具

目录 Postman测试工具接口测试工具swaggerKnife4j1.引入依赖2.配置3.常用注解4.接口测试 JMeter什么是JMeter?JMeter安装配置1.官网下载2.下载后解压3.汉语设置 JMeter的使用方法1.新建线程组2.设置参数3.添加取样器4.设置参数&#xff1a;协议&#xff0c;ip&#xff0c;端口…

【JUC】线程池ThreadPoolTaskExecutor与面试题解读

1、ThreadPoolTaskExecutor 创建线程池 从它的创建和使用说起&#xff0c;创建和使用的代码如下&#xff1a; 创建&#xff1a; ThreadPoolTaskExecutor executor new ThreadPoolTaskExecutor();executor.setCorePoolSize(corePoolSize);executor.setMaxPoolSize(maxPoolSize…

使用PHP实现随机调用图片

&#x1f482; 个人网站:【工具大全】【游戏大全】【神级源码资源网】&#x1f91f; 前端学习课程&#xff1a;&#x1f449;【28个案例趣学前端】【400个JS面试题】&#x1f485; 寻找学习交流、摸鱼划水的小伙伴&#xff0c;请点击【摸鱼学习交流群】 预览地址&#xff1a;ht…

【CI/CD】基于 Jenkins+Docker+Git 的简单 CI 流程实践(上)

基于 JenkinsDockerGit 的简单 CI 流程实践&#xff08;上&#xff09; 在如今的互联网时代&#xff0c;随着软件开发复杂度的不断提高&#xff0c;软件开发和发布管理也越来越重要。目前已经形成一套标准的流程&#xff0c;最重要的组成部分就是 持续集成 及 持续交付、部署。…

深入探索JavaEE单体架构、微服务架构与云原生架构

课程链接&#xff1a; 链接: https://pan.baidu.com/s/1xSI1ofwYXfqOchfwszCZnA?pwd4s99 提取码: 4s99 复制这段内容后打开百度网盘手机App&#xff0c;操作更方便哦 --来自百度网盘超级会员v4的分享 课程介绍&#xff1a; &#x1f50d;【00】模块零&#xff1a;开营直播&a…

2023-08-15 linux mipi 屏幕调试:有一个屏幕开机时候不显示,开机后按power 按键休眠唤醒就可以显示。原因是reset gpio 被复用

一、现象&#xff1a;今天更新了一个新版本的buildroot linux sdk &#xff0c;调试两个mipi 屏幕&#xff0c;这两个屏幕之前在其他的sdk都调好了的&#xff0c;所有直接把配置搬过来。但是有一个屏幕可以正常显示&#xff0c;有一个屏幕开机时候不显示&#xff0c;开机后按po…

django-基本环境配置

文章目录 django 环境安装1. 安装环境1.1 安装 Python (配置虚拟环境)1.1.1 步骤 1.2 Conda配置环境参考 django 环境安装 1. 安装环境 1.1 安装 Python (配置虚拟环境) 由于国外源速度慢&#xff0c;可以pip添加清华源 pip config set global.index-url https://pypi.tuna.…

Jmeter进阶使用:BeanShell实现接口前置和后置操作

一、背景 我们使用Jmeter做压力测试或者接口测试时&#xff0c;除了最简单的直接对接口发起请求&#xff0c;很多时候需要对接口进行一些前置操作&#xff1a;比如提前生成测试数据&#xff0c;以及一些后置操作&#xff1a;比如提取接口响应内容中的某个字段的值。举个最常用…

Mysql之explain详解

1. explain作用 使用explain可以展示出sql语句的执行计划&#xff0c;再根据sql的执行计划去判断这条sql有哪些点可以进行优化&#xff0c;从而让sql的效率达到最大化。 2. 执行计划各列含义 &#xff08;1&#xff09;id&#xff1a;id列是select的序列号&#xff0c;这个…

关于APP备案、小程序备案的问题,如何备案?

近日&#xff0c;工信部发布了关于开展移动互联网应用程序备案工作的通知。为落实相关法律法规要求&#xff0c;促进互联网行业规范健康发展&#xff0c;进一步做好移动互联网信息服务管理&#xff0c;现组织开展移动互联网应用程序&#xff08;以下简称 APP&#xff09;备案工…

数据结构的图存储结构

目录 数据结构的图存储结构 图存储结构基本常识 弧头和弧尾 入度和出度 (V1,V2) 和 的区别,v2> 集合 VR 的含义 路径和回路 权和网的含义 图存储结构的分类 什么是连通图&#xff0c;&#xff08;强&#xff09;连通图详解 强连通图 什么是生成树&#xff0c;生…

计算机竞赛 opencv 图像识别 指纹识别 - python

0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于机器视觉的指纹识别系统 &#x1f947;学长这里给一个题目综合评分(每项满分5分) 难度系数&#xff1a;3分工作量&#xff1a;3分创新点&#xff1a;4分 该项目较为新颖&#xff0c;适…

[oneAPI] 手写数字识别-GAN

[oneAPI] 手写数字识别-GAN 手写数字识别参数与包加载数据模型训练过程结果 oneAPI 比赛&#xff1a;https://marketing.csdn.net/p/f3e44fbfe46c465f4d9d6c23e38e0517 Intel DevCloud for oneAPI&#xff1a;https://devcloud.intel.com/oneapi/get_started/aiAnalyticsToolki…

【STM32】 工程

&#x1f6a9; WRITE IN FRONT &#x1f6a9; &#x1f50e; 介绍&#xff1a;"謓泽"正在路上朝着"攻城狮"方向"前进四" &#x1f50e;&#x1f3c5; 荣誉&#xff1a;2021|2022年度博客之星物联网与嵌入式开发TOP5|TOP4、2021|2022博客之星TO…

html2canvas生成图片地址Base64格式转成blob在转成file(二进制)可正常发送(保姆教程,复制粘贴可用)

开始: 最终结果: 1. html2canvas方法生成的图片地址已Base64编码形式放在img标签src中可直接展示生成的图片(注意页面标签获取位置,还有个setTimeout页面渲染需要时间) setTimeout(function () {var result {};v…

DiffusionDet: Diffusion Model for Object Detection

DiffusionDet: Diffusion Model for Object Detection 论文概述不同之处整体流程 论文题目&#xff1a;DiffusionDet: Diffusion Model for Object Detection 论文来源&#xff1a;arXiv preprint 2022 论文地址&#xff1a;https://arxiv.org/abs/2211.09788 论文代码&#xf…

24、springboot的自动配置01--类条件注解@ConditionalOnClass、bean条件注解@ConditionalOnBean

条件注解的理解&#xff1a;该注解指定了一些条件&#xff0c;只有符合这些条件&#xff0c;被该注解修饰的类或方法才能生效。 这些条件可以是yml配置文件里面的属性等数据是否存在&#xff0c;也可以是一些依赖驱动是否存在的条件、也可以是指定的bean是否存在等。 springbo…

Golang协程,通道详解

进程、线程以及并行、并发 关于进程和线程 进程&#xff08;Process&#xff09;就是程序在操作系统中的一次执行过程&#xff0c;是系统进行资源分配和调度的基本单位&#xff0c;进程是一个动态概念&#xff0c;是程序在执行过程中分配和管理资源的基本单位&#xff0c;每一…

iTOP-RK3588开发板安装TFTP服务端

首先在 ubuntu 中执行以下命令安装 TFTP 服务&#xff1a; apt-get install tftp-hpa tftpd-hpa 安装完成以后创建 TFTP 服务器工作目录,并对 TFTP 的服务配置文件进行修改,具体步骤如下&#xff1a; 输入以下命令在家目录创建 tftpboot 文件夹&#xff0c;如下图所示&#x…

Prompt、RAG、微调还是重新训练?如何选择正确的生成式AI的使用方法

生成式人工智能正在快速发展&#xff0c;许多人正在尝试使用这项技术来解决他们的业务问题。一般情况下有4种常见的使用方法&#xff1a; Prompt EngineeringRetrieval Augmented Generation (RAG 检索增强生成)微调从头开始训练基础模型(FM) 本文将试图根据一些常见的可量化…