首页 > 其他分享 >数据结构之环形队列

数据结构之环形队列

时间:2022-10-30 10:02:03浏览次数:66  
标签:队列 元素 环形 插入 int maxSize front 数据结构


概述

队列是一种具有先进先出(FIFO)的数据类型,可以使用多种数据结构来实现队列:数组和链表。

简单队列的应用场景比较有限,于是那些牛人们就发明一些复杂的队列:

  • 环形队列
  • 双端队列
  • 优先队列

应用场景

  • Memory Management: The unused memory locations in the case of ordinary queues can be utilized in circular queues.
  • Traffic system: In computer controlled traffic system, circular queues are used to switch on the traffic lights one by one repeatedly as per the time set.
  • CPU Scheduling: Operating systems often maintain a queue of processes that are ready to execute or that are waiting for a particular event to occur.
  • Lock Free Queue: When high performance is required, we don’t want to use lock. Circular Queue can is the top 1 data structure for lock free queue.

一般应用于需要高效且频繁进行多线程通信传递数据的场景,例如:Linux捕包、发包等等,(Linux系统中对​​PACKET_RX_RING​​​和​​PACKET_TX_RING​​的支持实质就是内核实现的一种环形队列),或者用于在进程间的异步数据传输或记录日志文件时,为解决某些特殊情况下的竞争问题提供一种免锁的方法。

分析

Circular Queue,环形缓冲区,RingBuffer,先进先出,提供对缓冲区的互斥访问,通常有一个读指针和写指针。实际环形队列在工作时有3种情况:

  1. 入队速度=出队速度
    一般情况,即入队速度和出队速度大致一样,即使某个突然时刻入队速度陡然变高或者出队速度陡然变低,都能通过队列这个缓冲区把这些数据先存起来,等到能处理的时候再处理。
  2. 入队速度>出队速度
    队列写入的速度>读取的速度,一段时间后,队列中大多数全是写入但没读取的元素,当又一个新的元素产生时,可以把这个新元素drop掉或者放在另一个缓冲区保存起来,此时说明你需要对出队处理元素的算法或逻辑优化处理速度。
  3. 入队速度<出队速度
    队列读取速度>写入速度,说明程序出队处理元素的速度很快,这是比较好的情况,不足之处:读取队列时可能经常会轮询队列是否有新的元素,造成cpu占用过高。

引申

引入环形队列,为了解决什么问题??

顺序队列的假溢出问题:队列的存储空间未满,却发生溢出。比如 rear 现在虽然指向最后一个位置的下一位置,但是之前队头也删除一些元素,队头指针经历若干次的 +1 之后,遗留下很多空位置,但是顺序队列还在傻乎乎的以为再有元素入队,就溢出呢!肯定不合理。故循环队列诞生!

有两种可行的解决方法:

  1. 平移元素:把元素平移到队列的首部。效率低。否决
  2. 将新元素插入到第一个位置上,构成循环队列,入队和出队仍按先进先出的原则进行。操作效率高、空间利用率高。

使用循环队列解决假溢出问题,但引入新问题:判空。

满足​​front = rear​​条件的队列有两种情况:队列空和队列满:

数据结构之环形队列_出队


数据结构之环形队列_数据_02


解决办法:

  1. 另设一个布尔变量以区别队列的空和满;
  2. 少用一个元素的空间,约定入队前测试尾指针在循环下加 1 后是否等于头指针,若相等则认为队满;(常用)
  3. 使用一个计数器记录队列中元素的总数。
实现

C参考实现​​GitHub​

环形队列可使用数组或链表来实现,一般很少用链表来实现,因为访问链表需要挨个遍历。

需要维护两个属性,即2个index,一个是写入index,标示当前可以写入元素的index,入队时使用。一个是读取index,标示当前可以读取元素的index,出队时使用。

另外,节点状态的切换也是一个问题,即当前节点是可读还是可写。解决方案:在队列中每个元素的头部加一个元素标示字段,标示这个元素是可读还是可写,而这个的关键就在于何时设置元素的可读可写状态,当这个元素读取完之后,要设置可写状态,当这个元素写入完成之后,要设置可读状态。

public class RingBuffer {
/**
* 最大总量
*/
private final int maxSize;

/**
* 表示队列的第一个元素的位置,初始值默认为0
*/
private int front;

/**
* 表示队列的最后一个元素的后一个位置,初始值默认为0,需要空出一个空间作为“约定”
*/
private int rear;

/**
* 存放数据用的数组队列
*/
private final int[] queueArray;

/**
* 初始化一个maxSize长度的数组队列。
*/
public RingBuffer(int maxSize) {
this.maxSize = maxSize;
this.queueArray = new int[maxSize];
}

/**
* 判断队列是否满
*/
public boolean isFull() {
return (rear + 1) % maxSize == front;
}

/**
* 判断队列是否为空
*/
public boolean isEmpty() {
return rear == front;
}

/**
* 往队列中添加一个元素
*/
public boolean addQueue(int element) {
if (isFull()) {
return false;
}
// 队尾后移一位
queueArray[rear] = element;
rear = (rear + 1) % maxSize;
return true;
}

/**
* @return 弹出&获取当前队首任务
*/
public int takeQueue() {
if (isEmpty()) {
throw new RuntimeException("队列为空");
}
// 队首后移一位
int value = queueArray[front];
queueArray[front] = 0;
front = (front + 1) % maxSize;
return value;
}

/**
* 显示队列里的数据
*/
public void showQueue() {
if (isEmpty()) {
System.out.println("队列里没有数据");
return;
}
for (int i = front; i < front + size(); i++) {
System.out.println("当前数据顺序位数 = " + queueArray[i]);
}
}

/**
* @return 队列中有效数据个数
*/
public int size() {
return (rear + maxSize - front) % maxSize;
}

public static void main(String[] args) {
int size = 10;
RingBuffer ringBuffer = new RingBuffer(size);
for (int i = 0; i < size; i++) {
boolean success = ringBuffer.addQueue(i);
System.out.println(success ? "插入" + i + "成功" : "插入" + i + "失败");
}
for (int i = 0; i < size + 1; i++) {
if (i == size - 1) {
int item = ringBuffer.takeQueue();
System.out.println("取出" + item + "成功");
}
if (i == size) {
int item = ringBuffer.takeQueue();
System.out.println("取出" + item + "成功");
}
boolean putSuccess = ringBuffer.addQueue(i);
// 只有在i=9和10时才会各读取一次,所以这里的插入大部分都会失败,因为前面写入一轮缓冲区已经满
System.out.println(putSuccess ? "插入" + i + "成功" : "插入" + i + "失败");
}
}

}

输出:

插入0成功
插入1成功
插入2成功
插入3成功
插入4成功
插入5成功
插入6成功
插入7成功
插入8成功
插入9失败
插入0失败
插入1失败
插入2失败
插入3失败
插入4失败
插入5失败
插入6失败
插入7失败
插入8失败
取出0成功
插入9成功
取出1成功
插入10成功
Process finished with exit code 0
参考

队列的存储结构和常见操作(C语言实现)
无锁环形队列的一种高效实现
环形队列




标签:队列,元素,环形,插入,int,maxSize,front,数据结构
From: https://blog.51cto.com/u_15851118/5807205

相关文章

  • 驱动开发之基本数据结构
    根据MSDN的介绍,自己对一些基本结构做一些翻译,帮助自己理解。驱动对象 DRIVER_OBJECTtypedefstruct_DRIVER_OBJECT{CSHORTType;CSHORT......
  • 数据结构—第二章线性表习题
    (1)B(2)A(3)B(4)A(5)D(6)B(7)C(8)A(9)B(10)D(11)C(12)D(13)D(14)A(15)C(1)voidMergeList(LinkList&La,LinkList&Lb,LinkList&Lc){//将两个递增的有序链表La和Lb合并为一个递增的有序链表Lc......
  • 20221027数据结构与算法之线性表——顺序表
    广州疫情被封区,在家学习#pragmawarning(disable:4996)#include<stdio.h>#include<stdlib.h>//动态顺序表的实现typedefintdata_t;typedefstructSeqList{data_t*da......
  • 数据结构 玩转数据结构 4-5 从链表中删除元素
    0课程地址https://coding.imooc.com/lesson/207.html#mid=13448 1重点关注1.1代码草图解析    2课程内容3Codi......
  • 【数据结构-数组】数组的基本操作
    目录1数据结构定义2插入操作3删除操作4按值查找1数据结构定义#defineMAX50typedefstruct{intdata[MAX];intlength;}SqList;初始化:voidIni......
  • 使用数据结构中的队列解决舞伴搭配问题
    ​ (一)问题描述某班有m个女生,n个男生(m不等于n,男女生人数和不能小于20),现要举办一个舞会,男女生分别编号坐在舞池两边的椅子上等待。每曲开始时,依次从男生和女生中各出一......
  • 数据结构 树(第10-14天)
    树的题目太多了,先总结一下树的遍历方式。按照根节点的遍历顺序。可以分为前序、中序、后序。前序遍历,即根–>左–>右的顺序。中序遍历,左–>根–>右。后续遍历,左–>右–>......
  • 数据结构 栈 / 队列(第9天)
    20.有效的括号判断输入的括号是否有效。左右括号··能闭合,顺序合适。思路:用栈实现。遇到左括号就保存在栈中,遇到右括号则需要从栈中弹出一个括号,与之配对。classSolutio......
  • 数据结构 链表(第7、8天)
    链表这里面的链表题比较简单,只要会遍历链表、删除链表节点、反转链表这些基本操作就行。必要时可以画图辅助理解。141.环形链表给定一个链表,判断是否有环。思路:快慢指针。......
  • 数据结构模板整合
    栈#include<iostream>#include<cstdio>#definemaxn100010usingnamespacestd;intn,x;template<typenametype>inlinevoidread(type&x){x=0;boolfl......