数据结构--队列与循环队列

这篇具有很好参考价值的文章主要介绍了数据结构--队列与循环队列。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

队列

        队列是什么,先联想一下队,排队先来的人排前面先出,后来的人排后面后出;队列的性质也一样,先进队列的数据先出,后进队列的后出;就像图一的样子:数据结构--队列与循环队列,数据结构

 图1

        如图1,1号元素是最先进的,开始出队时,那么他就是最先出的,然后12进队,就应该排在最后面,等待前面的所有元素出队完成后才能出队;有个专业的名词叫FIFO(first in first out),翻译过来就是先进先出的意思;

队列的数据结构:

        数据结构 = 结构定义 + 结构操作;

        队列的结构定义就是:

       物理结构:

        一个存储数据的数据域,这里我们用的是数组;一个头指针,一个尾指针;头指针指向,下个出队的元素的位置,尾指针指向最后一个元素的位置,然后还有队列的长度,元素个数;

        那么用结构体封装他的物理结构代码如下:

typedef struct Queue {
    int size, cnt, head, tail;//4个变量分别是,队列长度,元素个数,头指针,尾指针
    //因为用的是数组,所以头尾指针,直接用一个int变量就可以存贮了
    void *data;//数据域
} Queue;

        逻辑结构:

        他的逻辑结构就是,先进先出,需要去维护这个性质,如果破坏了性质就不能算做数据结构了,因为你破坏了它的结构定义;所以一定不要破坏数据结构的结构定义;

结构操作:

        说完了结构定义来看下,队列它是如何出队入队的:

数据结构--队列与循环队列,数据结构

        现在出队一个元素,那么Head指针就应该指向下一个位置,也就是位置1,那么head++,head = 1:

数据结构--队列与循环队列,数据结构

         现在入队一个元素,假如入队元素12,那么Tail指针应该先Tail++,在放入新元素,不然就覆盖掉了元素11,如下图:

数据结构--队列与循环队列,数据结构

数据结构--队列与循环队列,数据结构

        可能有人会问,1不是出队了嘛,为什么在图中还有,是我画图没有画完,但是,在写代码的时候,情况就是这样的,因为这是一个数组,你只是吧头指针往后偏移了,但是那个位置的元素他还是存在的,只是不会去访问到了,那么他也相当于出队了;也就是相当于我们在数组上面维护了一个队列,他从头部减少,尾部增加的一个思想;

循环队列

        提到队列了,也不得不提循环队列,循环队列是什么,假如长度为10的队列,它入队了10个元素,也出队了10个元素,那么头尾指针现在是在同一个位置,就是下图情况:

数据结构--队列与循环队列,数据结构

        他现在里面是没有元素的,你现在看到的1-9是已经出队了的,10是还没有出队的,那么怎么办,那就直接让tail = 0,又从数组的头部开始 ,如图

        数据结构--队列与循环队列,数据结构

         元素11入队,直接覆盖掉之前的元素1,那么下次入队就是从位置1开始,出队还是元素10先出队,然后出队后,Head指针那么也应该等于0,也从数组的头开始再次出队;

        那么如何去判断队列为空呢,在定义物理结构是吗,有一个变量记录着,队列当前的元素个数;

代码实现:

        那么思路大概讲完了,代码实现的是循环队列,来看代码实现:

        

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

typedef struct Queue {
    int size, cnt, head, tail;//4个变量分别是,队列长度,元素个数,头指针,尾指针
    //因为用的是数组,所以头尾指针,直接用一个int变量就可以存贮了
    int *data;//数据域
} Queue;

Queue *init(int n) {//初始化队列,向计算机借空间
    Queue *q = (Queue *)malloc(sizeof(Queue));
    q->data = (int *)malloc(sizeof(int) * n);
    q->size = n;
    q->cnt = q->head = q->tail = 0;
    return q;
}
int empty(Queue *);
int front(Queue *q) {//获取队列头部元素
    if(empty(q)) return -1;
    return q->data[q->head];
}

int empty(Queue *q) {//判读队列是否为空
    return q->cnt == 0;
}

int push(Queue *q, int val) {//入队
    if (q->cnt == q->size) return 0;
    q->data[q->tail++] = val;
    if (q->tail == q->size) q->tail = 0;
    q->cnt++;
    return 1;
}

int pop(Queue *q) {//出队
    if (empty(q)) return 0;
    q->head++;
    q->cnt--;
    if (q->head == q->size) q->head = 0;
    return 1;
}
void clear(Queue *q) {//有借有还
    if (!q) return ;
    free(q->data);
    free(q);
    return ;
}

void output(Queue *q) {//打印队列中的元素
    printf("Queue(%d) :[", q->cnt);
    for (int i = q->head, j = 0; j < q->cnt; j++) {
        j && printf(" ");
        printf("%d", q->data[(i + j) % q->size]);
    }
    printf("]\n");
    return ;
}

int main() {//测试
    srand(time(0));
    int op, val;
    Queue *q = init(5);
    for (int i = 0; i < 20; i++) {
        op = rand() % 4;         
        val = rand() % 100;
        switch (op) {
            case 0:
            case 1:
            case 2: {
                printf("%d push in Queue is %d\n", val, push(q, val));
            } break;
            case 3: {
                printf("%d ", front(q));
                printf("pop Queue is %d\n", pop(q));
            } break;
        }
        output(q);
    }
    clear(q);
    return 0;
}

 文章来源地址https://www.toymoban.com/news/detail-691392.html

到了这里,关于数据结构--队列与循环队列的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 数据结构--循环队列、链队

    //循环队列数据结构 typedef struct { QElemType data[MaxQSize];//数据域 int front,rear; //队头队尾指针 }SqQueue; //链队结点数据结构 typedef struct QNode { int data;//数据域 struct QNode* next;//指针域 }QNode, * QueuePtr; typedef struct { struct QNode* front, * rear;//rear指针指向队尾 用于入队 front指针指向队头 用于

    2024年02月15日
    浏览(30)
  • 数据结构——循环队列详解

    目录 一、循环队列的定义 二、 循环队列的基本操作 三、循环队列的实现  1、循环队列的定义 2、循环队列的初始化  3、循环队列出队  4、循环队列入队  5、队列判空 6、 队列判满 7、取队头元素 8、输出队列  9、求队列长度  四、完整代码  五、小结  六、参考文献 一、

    2024年02月04日
    浏览(29)
  • 数据结构——循环队列的实现

    hello hello~ ,这里是大耳朵土土垚~💖💖 ,欢迎大家点赞🥳🥳关注💥💥收藏🌹🌹🌹 💥 个人主页:大耳朵土土垚的博客 💥 所属专栏:数据结构学习笔记 💥对于数据结构顺序表、链表、堆有疑问的都可以在上面数据结构的专栏进行学习哦~ 有问题可以写在评论区或者私信

    2024年03月24日
    浏览(30)
  • 九、数据结构——顺序队列中的循环队列

    一、循环队列的定义 二、循环队列的实现 三、循环队列的基本操作 ①初始化 ②判空 ③判满 ④入队 ⑤出队 ⑥获取长度 ⑦打印 四、循环队列的应用 五、全部代码 在数据结构中,队列(Queue)是一种常见的线性数据结构,遵循先进先出(First In First Out,FIFO)的原则。循环队

    2024年02月15日
    浏览(34)
  • 数据结构第九弹---循环队列

    顺序队列在使用过程中容易出现虚假的满状态, 为了解决这个问题,就产生了一个较巧妙的方法,将顺序队列臆造为一个环状的空间,称之为循环队列。循环队列中指针和队列元素之间的关系不变,我们只需要利用模运算就可以很容易实现指针的循环移动。但是循环队列中存

    2024年01月16日
    浏览(29)
  • 数据结构OJ:设计循环队列

    本题为LeetCode上的经典题目,题目要求我们设计一种循环队列,满足FIFO原则且队尾被连接在队首之后。 题目中介绍循环队列的好处是可以重复利用空间,所以我们很容易想到在初始化时即开辟指定大小的空间,之后便不需要再开辟空间,只需后续销毁即可。 首先我们要选择

    2024年04月17日
    浏览(49)
  • 【数据结构与算法】设计循环队列

      🧑‍🎓 个人主页:简 料   🏆 所属专栏:C++   🏆 个人社区:越努力越幸运社区   🏆 简       介: 简料简料,简单有料~在校大学生一枚,专注C/C++/GO的干货分享,立志成为您的好帮手 ~ C/C++学习路线 (点击解锁) ❤️ C语言阶段(已结束) ❤️ 数据结构与算法(ing) ❤

    2024年01月17日
    浏览(33)
  • 王道考研数据结构--4.2循环队列

    目录 前言  1.循环队列的定义 2.循环队列的结构 3.循环队列的操作 3.1定义循环队列 3.2初始化 3.3入队 3.4出队 3.5遍历,求表长 3.6清空销毁 4.完整代码 日期:2023.7.25 书籍:2024年数据结构考研复习指导(王道考研系列) 内容:实现顺序队列的基本实现,主要功能如下: 1.循环队

    2024年02月15日
    浏览(35)
  • 【数据结构】队列的使用|模拟实现|循环队列|双端队列|面试题

    1.1 概念 队列 :只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出FIFO(First In First Out) 入队列:进行插入操作的一端称为 队尾(Tail/Rear) 出队列:进行删除操作的一端称为 队头(Head/Front) 队列和栈的区别: 队列是 先进先出(队

    2024年02月03日
    浏览(32)
  • 【数据结构与算法】03 队列(顺序队列--循环队列--优先级队列--链队列)

    队列( queue )是一种常见的数据结构,它遵循先进先出(FIFO)的原则。队列可以理解为一个具有两个端点的线性数据结构,其中一个端点称为\\\"队尾\\\"(rear),用于插入新元素,另一个端点称为\\\"队首\\\"(front),用于移除元素。新元素被插入到队尾,而最早插入的元素总是在队

    2024年02月08日
    浏览(43)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包