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

  • (꒪ꇴ꒪ ),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;}//先挪rearque->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;}//先挪frontque->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;}//先挪rearque->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;}//先挪frontque->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;//尾部节点要变成新节点newq->rear = new;}//队的元素个数要+1q->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;//链式队列的元素个数-1q->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)开发板实战

📢写在最后

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

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言库函数之 qsort 讲解、使用及模拟实现

引入 我们在学习排序的时候&#xff0c;第一个接触到的应该都是冒泡排序&#xff0c;我们先来复习一下冒泡排序的代码&#xff0c;来作为一个铺垫和引入。 代码如下&#xff1a; #include<stdio.h>void bubble_sort(int *arr, int sz) {int i 0;for (i 0; i < sz…

使用navicat连接postgresql报错问题解决

使用navicat连接postgresql报错问题解决 一、问题现象&#xff1a; 最近使用Navicat来连接postgreSQL数据库&#xff0c;发现连接不上&#xff0c;报错信息如下&#xff1a; 自己百度了一下&#xff0c;发现pgsql 15版本以后&#xff0c;有些系统表的列名改了&#xff0c;pg_…

【JavaSE】面向对象之继承

继承 继承概念继承的语法父类成员的访问子类和父类没有同名成员变量子类和父类有同名成员变量成员方法名字不同成员方法名字相同 super关键字子类构造方法super和this继承方式 继承概念 继承(inheritance)机制&#xff1a;是面向对象程序设计使代码可以复用的最重要的手段&…

《Go 语言第一课》课程学习笔记(四)

构建模式&#xff1a;Go Module 的 6 类常规操作 为当前 module 添加一个依赖 我们如何为一个 Go Module 添加一个新的依赖包呢&#xff1f; 如果我们要为项目增加一个新依赖&#xff1a;github.com/google/uuid&#xff0c;我们首先会更新源码&#xff1a;package mainimpor…

【100天精通python】Day37:GUI界面编程_PyQT从入门到实战(上)

目录 专栏导读 1 PyQt6 简介&#xff1a; 1.1 安装 PyQt6 和相关工具&#xff1a; 1.2 PyQt6 基础知识&#xff1a; 1.2.1 Qt 的基本概念和组件&#xff1a; 1.2.2 创建和使用 Qt 窗口、标签、按钮等基本组件 1.2.3 布局管理器&#xff1a;垂直布局、水平布局、网格布局…

uniapp开发微信小程序使用painter将页面转换为图片并保存到本地相册

引言 我使用到painter的原因是&#xff0c;在uniapp开发微信小程序时&#xff0c;需要将一个页面的内容转换成图片保存到本地相册。 起初在网上找到很多都是在uniapp中使用 html2canvas 将网页转换成图片再jspdf将图片转换为pdf&#xff0c;但是这种方式在小程序环境不支持&am…

Leetcode151 翻转字符串中的单词

给你一个字符串 s &#xff0c;请你反转字符串中 单词 的顺序。 单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开。 返回 单词 顺序颠倒且 单词 之间用单个空格连接的结果字符串。 注意&#xff1a;输入字符串 s中可能会存在前导空格、尾随空格…

最新ChatGPT网站AI系统源码+详细图文搭建教程/支持GPT4.0/AI绘画/H5端/Prompt知识库/

一、前言 SparkAi系统是基于国外很火的ChatGPT进行开发的Ai智能问答系统。本期针对源码系统整体测试下来非常完美&#xff0c;可以说SparkAi是目前国内一款的ChatGPT对接OpenAI软件系统。 那么如何搭建部署AI创作ChatGPT&#xff1f;小编这里写一个详细图文教程吧&#xff01…

小红书美妆护肤种草推广:深度剖析与实战策略

在这个平台上&#xff0c;用户可以分享自己的购物心得和产品评价&#xff0c;为其他消费者提供购买参考。这种基于用户真实体验的分享&#xff0c;更容易赢得消费者的信任&#xff0c;从而提高产品的购买转化率。 小红书俨然成为了美妆护肤品牌种草推广的主要战场&#xff0c;…

Selenium浏览器自动化测试框架简单介绍

selenium简介 介绍   Selenium [1] 是一个用于Web应用程序测试的工具。Selenium测试直接运行在浏览器中&#xff0c;就像真正的用户在操作一样。支持的浏览器包括IE&#xff08;7, 8, 9, 10, 11&#xff09;&#xff0c;Mozilla Firefox&#xff0c;Safari&#xff0c;Googl…

Linux:如何挂载Window的共享目录

本文介绍的方法操作简单快捷&#xff0c;实用性强。下面就让小编来带大家学习“Linux下怎么挂载Window中的共享目录”吧! 一、在Window下创建共享目录 1、首先&#xff0c;在Window下创建一个目录作为共享目录&#xff0c;此处创建的目录名为ShareDir 2、右键目录&#xff0c…

3.微服务概述

1.大型网络架构变迁 SOA与微服务最大的差别就是服务拆分的细度&#xff0c;目前大多数微服务实际上是SOA架构&#xff0c;真正的微服务应该是一个接口对应一个服务器&#xff0c;开发速度快、成本高&#xff1b; 微服务SOA能拆分的就拆分是整体的&#xff0c;服务能放一起的都…