数据结构(顺序队列——c语言实现)

队列的概念:

       队列是限制在两端进行插入和删除操作的线性表,允许进行存入的一端称为“队尾”,允许进行删除操作的一端称为“队头”。当线性表中没有元素时,称为“空队”。特点:先进先出(FIFO)。

规定一:front指向队头元素的前一个位置;rear指向队尾元素所在位置。

规定二:front指向队头元素的位置;rear指向队尾元素的下一个位置。

        在队列操作过程中,为了提高效率,以调整指针代替队列元素的移动,并将数组作为循环队列的操作空间。

       为了区别空队和满队,满队元素个数比数组元素个数少一个。

两个规定二选其一遵守,要么遵守一,要么遵守二。

顺序队列的优缺点:

优点

  1. 存储效率高

           顺序队列使用连续的内存空间,内存利用率较高,不会有像链表那样的指针开销。
  2. 访问速度快

           由于元素在内存中是连续存储的,顺序队列在访问元素时具有较高的缓存命中率,因此访问速度较快。
  3. 实现简单

           顺序队列的实现相对简单,通常只需要一个数组和两个指针(front 和 rear)来管理队列的头和尾。
  4. 空间预分配

           可以在创建队列时预先分配一个固定大小的数组,避免了在队列动态增长时频繁的内存分配和释放操作。

缺点

  1. 固定大小

           顺序队列的大小在创建时是固定的,如果队列满了,就不能再插入新元素,除非进行额外的扩容操作。扩容操作通常涉及复制整个数组到新的内存空间,这可能导致性能下降。
  2. 假溢出

           在顺序队列中,即使数组还有空闲空间,如果所有的空间都在队列的前面(已经被已删除的元素占用),但后面的空间还未使用,也会导致队列无法插入新元素,这就是所谓的“假溢出”问题。
  3. 内存碎片

           在频繁的插入和删除操作后,顺序队列可能会产生内存碎片,导致内存使用效率下降。虽然这通常不是主要问题,但在某些情况下需要考虑。
  4. 扩展性较差

           如果队列的大小需要频繁变化,顺序队列的扩展性较差。相比之下,链表实现的队列(如链式队列)在动态扩展方面更具优势。

顺序队列

下面以遵守规则二为例来编写代码:

#ifndef _SEQQUEUE_H
#define _SEQQUEUE_H

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

typedef char Type;
#define LEN 10
//约定1:LEN-1号下标的下一个为0号下标
//约定2:数组少存一个数据,最多只存LEN-1个元素
typedef struct{
    Type data[LEN];   //队列空间
    int front;        //队头的下标
    int rear;         //队尾下标+1
}queue;

//创建
queue *create_queue();
//判空 空为0非空为1
int empty_queue(queue *q);
//判满 满为0不满为1
int full_queue(queue *q);
//入队enqueue
void enter_queue(queue *q,Type data);
//出队dequeue
Type delete_queue(queue *q);
//初始化
void init_queue(queue *q);
//回收
void free_queue(queue **q);

#endif

       顺序队列同样是使用数组来存储队列中的元素,结构体里有三个成员,一个是存放队列元素的数组,另外两个是队列数组两个下标;数组的长度用一个宏定义的LEN来代表,之后要修改队列长度直接修改宏定义即可。

#include "seqqueue.h"

//创建
queue *create_queue()
{
    queue *q = (queue *)malloc(sizeof(queue));
    if(NULL == q)
    {
        perror("create queue malloc");
        return NULL;
    }
    q->front = q->rear = 0;
    return q;
}
//判空 空为0,非空为1
int empty_queue(queue *q)
{
    if(NULL == q)
    {
        puts("queue is NULL");
        return -1;
    }
    if(q->front == q->rear)
    {
        puts("queue is empty");
        return 0;
    }
    else
        return 1;
}
//判满 满为0,不满为1
int full_queue(queue *q)
{
    if(NULL == q)
    {
        puts("queue is NULL");
        return -1;
    }
    if(q->front == (q->rear+1)%LEN)
    {
        puts("queue is full");
        return 0;
    }
    else
        return 1;
}
//入队enqueue
void enter_queue(queue *q,Type data)
{
    if(1 == full_queue(q))
    {
#if 1
        q->rear = q->rear%LEN; //让10变为0
        q->data[q->rear] = data;
        q->rear++;
#else
        q->data[q->rear] = data;
        q->rear = (q->rear+1)%LEN;
#endif
    }
}
//出队dequeue
Type delete_queue(queue *q)
{
    if(1 == empty_queue(q))
    {
        Type t = q->data[q->front];
        q->front = (q->front+1)%LEN;
        return t;
    }
}
//初始化
void init_queue(queue *q)
{
    if(NULL == q)
    {
        puts("queue is NULL");
        return;
    }
    q->rear = q->front;
}
//回收
void free_queue(queue **q)
{
    if(NULL == *q)
    {
        puts("queue is NULL");
        return;
    }
    free(*q);
    *q = NULL;
}

创建: queue *create_queue()

       在创建顺序队列的时候也是在堆区开辟空间,在这里一开始队列为空,所以这时候尾的下标应该为-1,头的下标为0,但是我们规定rear为尾+1,因此在创建完结构体之后,给front和rear都赋值为0即可。

判空:int empty_queue(queue *q)

       在前面的分析中我们已经知道了队列为空的时候front和rear的值应该相同,因此在这里只需要判断结构体里的front和rear是否相等就能判断队列是否为空,空函数返回0,非空函数返回1。

判满:int full_queue(queue *q)

       在判满这里分析中也提到了,为了避免判空和判满发生冲突,所以数组最多存储LEN-1个元素,但是除此之外,为了提高利用率,我们把数组作为循环队列来操作;在这里就会出现一个问题,那就是当队列放满,数据存储刚好到数组倒数第二位时,rear指向的就是数组最后一个元素的位置,这时rear+1就会超出数组下标的范围,此时front为数组第一个元素的下标,所以为了让rear+1与front相等,我们对rear+1进行除以LEN取余操作,这样当rear+1=LEN时,取余正好为0,就是第一个元素的下标;故判满的条件就为(q->front == (q->rear+1)%LEN

入队:void enter_queue(queue *q,Type data)

       入队是在队尾的位置进行操作,也就是下标为rear的位置,但是在入队操作之前我们需要做一个判断,如果队列是满的那就不能再进行入队操作,只有队列不满才能进行入队操作;在入队之前我们需要保证rear的值不超过LEN-1,也就是当数组最后一个位置放满尾就应该移动到数组开头,所以在这里就先让rear取余LEN,再把data放入队列,最后让rear往后移动一位即可。

出队:Type delete_queue(queue *q)

       出队我们需要拿到出队的这个成员的值,因此函数返回值为Type类型,出队之前同样也需要进行一个判断,如果队列是空的,那就不能进行出队操作;当队列非空时,操作队头进行出队操作,将要出队的元素用一个变量t保存,然后让队头下标front往后移动一位,在这里为了防止超出数组操作范围,也需要对front进行取余,保证它在0~9之间;最后再将t返回即可。

初始化:void init_queue(queue *q)

       初始化的意思也就是将队列恢复到最开始空的时候,对于顺序队列,它为空的时候是front=rear,因为是数组,因此赋值可以覆盖,所以初始化我们只需要让结构体里的队头下标front和rear相等即可,这样队列相当于就是空队列,后续赋值会直接覆盖,不会造成别的影响。

回收:void free_queue(queue **q)

       在这里传参同样需要传指针的地址,因为存数据的数组在结构体里面,故在回收的时候不需要进行初始化,直接将结构体空间回收,再将指针指向NULL即可。

测试(主函数)

#include "seqqueue.h"
int main(int agrc,char *agrv[])
{
    queue *q = create_queue();
    printf("空为0:%d\n",empty_queue(q));
    puts("A,B,C,D入队");
    enter_queue(q,'A');
    enter_queue(q,'B');
    enter_queue(q,'C');
    enter_queue(q,'D');

    puts("出队");
    printf("%c\n",delete_queue(q));
    printf("%c\n",delete_queue(q));

    printf("判空:%d\n",empty_queue(q));

    puts("初始化");
    init_queue(q);
    printf("判空:%d\n",empty_queue(q));

    puts("回收");
    free_queue(&q);
    printf("%p\n",q);
    return 0;
}

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

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

相关文章

3C产品说明书电子化转变:用户体验、环保与商业机遇的共赢

在科技日新月异的当代社会&#xff0c;3C产品&#xff08;涵盖计算机类、通信类和消费类电子产品&#xff09;已成为我们日常生活中不可或缺的重要元素。与此同时&#xff0c;这些产品的配套说明书也经历了一场从纸质到电子化的深刻变革。这一转变不仅体现了技术的飞速进步&…

web服务nginx实验6:nginx发布动态页面的方法

安装软件&#xff1a; 启动服务&#xff1a; 创建文件&#xff1a; 再vim打开&#xff0c;写东西&#xff1a; 重启服务&#xff1a; Windows客户端测试&#xff1a;&#xff08;服务端要关防火墙&#xff09; 删除默认访问发.php文件&#xff1a; 创建一个新的配置文件&#x…

Three.js 相机控制器Controls

在 3D 场景中&#xff0c;摄像机的控制尤为重要&#xff0c;因为它决定了用户如何观察和与场景互动。Three.js 提供了多种相机控制器&#xff0c;最常用的有 OrbitControls、TrackballControls、FlyControls 和 FirstPersonControls。OrbitControls 适合用于查看和检查 3D 模型…

应急响应:玄机_Linux后门应急

https://xj.edisec.net/challenges/95 11关做出拿到万能密码&#xff0c;ATMB6666&#xff0c;后面都在root权限下操作 1、主机后门用户名称&#xff1a;提交格式如&#xff1a;flag{backdoor} cat /etc/passwd&#xff0c;发现后门用户 flag{backdoor} 2、主机排查项中可以…

.NET 9与C# 13革新:新数据类型与语法糖深度解析

记录&#xff08;Record&#xff09;类型 使用方式&#xff1a; public record Person(string FirstName, string LastName); 适用场景&#xff1a;当需要创建不可变的数据结构&#xff0c;且希望自动生成 GetHashCode 和 Equals 方法时。不适用场景&#xff1a;当数据结构需…

阿里云IIS虚拟主机部署ssl证书

宝塔配置SSL证书用起来是很方便的&#xff0c;只需要在站点里就可以配置好&#xff0c;但是云虚拟主机在管理的时候是没有这个权限的&#xff0c;只提供了简单的域名管理等信息。 此处记录下阿里云&#xff08;原万网&#xff09;的IIS虚拟主机如何配置部署SSL证书。 进入虚拟…

Vue通过file控件上传文件到Node服务器

功能&#xff1a; 1.多文件同时上传、2.拖动上传、3.实时上传进度条、4.中断上传和删除文件、5.原生file控件的美化 搁置的功能: 上传文件夹、大文件切片上传、以及其他限制条件未处理 Node服务器的前置准备: 新建文件夹: file_upload_serve初始化npm: npm …

VSCode+ESP-IDF开发ESP32-S3-DevKitC-1(1)开发环境搭建

VSCodeESP-IDF开发ESP32-S3-DevKitC-1&#xff08;1&#xff09;开发环境搭建 1.开发环境搭建&#xff08;安装ESP-IDF&#xff09;2.开发环境搭建&#xff08;安装VS Code&#xff09;3.开发环境搭建&#xff08;VSCode中安装ESP-IDF插件及配置&#xff09; 1.开发环境搭建&am…

个人全栈开发微信小程序上线了(记日记)

个人开发的全栈项目&#xff0c;《每日记鸭》微信小程序上线了&#xff01; 主要是技术栈&#xff1a;uniapp,koa2,mongodb,langchian&#xff1b; 感兴趣的小伙伴可以来捧捧场&#xff01;

南京邮电大学算法设计-二叉树先序遍历算法动态演示

二叉树先序遍历算法动态演示 一、课题内容和要求 (1)实验目的&#xff1a; 本实验通过手动输入二叉树结点信息&#xff0c;构建相应的二叉树&#xff0c;并通过图形化界面动态演示先序遍历算法的过程。通过本次实验&#xff0c;我可以深入理解二叉树的数据结构、先序遍历算法…

IntelliJ+SpringBoot项目实战(十)--常量类、自定义错误页、全局异常处理

一、常量类 在项目开发中&#xff0c;经常需要约定一些常量&#xff0c;比如接口返回响应请求指定状态码、异常类型、默认页数等&#xff0c;为了增加代码的可阅读性以及开发团队中规范一些常量的使用&#xff0c;可开发一些常量类。下面有3个常量类示例&#xff0c;代码位于op…

2025蓝桥杯(单片机)备赛--扩展外设之DS1302的使用(九)

1.DS1302数据手册的使用 a. DS1302 features: 工作电压&#xff1a;2V-5.5V 通信协议&#xff1a;3线接口&#xff08;CE、IO、SCLK&#xff09; 计时&#xff1a;秒、分、小时、月日期、月、星期、年&#xff08;闰年补偿器期至2100年&#xff09; b.原理图接线说明&#xff…

MCU中的定时器

第一章 定时器的应用场景 第二章 定时器的原理 2.1 定时器的计数原理 1. 定时器的本质是一个计数器&#xff1b; 2. 计数器是对输入的系统频率信号进行计数&#xff1b; 3. 每来一个周期的信号&#xff0c;计数器的cnt 加一。如果周期T表示为1s&#xff0c;来三个周期就表示…

类和对象——static 成员,匿名对象(C++)

1.static成员 a&#xff09;⽤static修饰的成员变量&#xff0c;称之为静态成员变量&#xff0c;静态成员变量⼀定要在类外进行初始化。 b&#xff09;静态成员变量为所有类对象所共享&#xff0c;不属于某个具体的对象&#xff0c;不存在对象中&#xff0c;存放在静态区。 …

POD-Transformer多变量回归预测(Matlab)

目录 效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现POD-Transformer多变量回归预测&#xff0c;本征正交分解数据降维融合Transformer多变量回归预测&#xff0c;使用SVD进行POD分解&#xff08;本征正交分解&#xff09;&#xff1b; 2.运行环境Matlab20…

C#中的二维数组的应用:探索物理含义与数据结构的奇妙融合

在C#编程中&#xff0c;二维数组&#xff08;或矩阵&#xff09;是一种重要的数据结构&#xff0c;它不仅能够高效地存储和组织数据&#xff0c;还能通过其行、列和交叉点&#xff08;备注&#xff1a;此处相交处通常称为“元素”或“单元格”&#xff0c;代表二维数组中的一个…

【网络安全 | 漏洞挖掘】通过密码重置污染实现账户接管

未经许可,不得转载。 文章目录 密码重置污染攻击漏洞挖掘的过程目标选择与初步测试绕过 Cloudflare 的尝试发现两个域名利用 Origin 头部污染实现账户接管攻击流程总结在今天的文章中,我们将深入探讨一种 账户接管 漏洞,并详细分析如何绕过 Cloudflare 的保护机制,利用密码…

uniapp 相关的swiper的一些注意事项

先推荐一个一个对标pc端swiper的uniapp版本 zebra-swiper 缺点是自定义分页器不是很好处理 不知道怎么弄 优点:可以进行高度自适应 &#xff08;这个uniapp原生swiper没有 只能动态修改 采用js 或者只有几种固定高度时采用变量修改&#xff09; <swiperref"lifeMiddle…

机器学习笔记——聚类算法(Kmeans、GMM-使用EM优化)

本笔记介绍机器学习中常见的聚类算法&#xff08;Kmeans、GMM-使用EM优化&#xff09;。 文章目录 聚类K-Means工作原理特点 K-Medoids工作原理特点 Mini-Batch K-Means工作原理特点 K-Means&#xff08;重要&#xff09;工作原理特点 总结K的选值1. 肘部法则&#xff08;Elbow…

SpringBoot项目升级到3.*,并由JDK8升级到JDK21

文章目录 技术选型说明JDK21的Demo项目下载升级过程出现的问题及解决1、程序包javax.servlet.http不存在1.1、java.lang.NoClassDefFoundError: javax/xml/bind/DatatypeConverter1.2、javax.validation包替换为jakarta.validation1.3、jakarta的名字由来 2、mybatis-plus升级3…