C语言实现循环队列

C语言实现循环队列详解循环队列的巧妙之处

大家好,又见面了,我是你们的朋友全栈君。

顺序队列的假溢出

顺序队列假溢出

1️⃣:初始化空队列,q -> front = q -> rear = 0

2️⃣:入队a1、a2、a3,q -> front = 0, q -> rear = 3

3️⃣:出队a1、a2,q -> front = 2, q -> rear = 3

4️⃣:入队a4、a5、a6,q -> front = 2, q -> rear = 6

执行 4️⃣ 操作后队列的”尾指针”已经超出对队列的最大范围了,之后无法再进行入队操作,而队列实际空间并未占满,就造成了假溢出。

解决上溢的方法

1、将队中元素依次向队头方向移动。

  • 缺点:浪费时间。每移动一次, 队中元素都要移动

2、将队列空间设想成一个循环的表,即分配给队列的 m 个存储单元可以循环使用,当 rearMAXSIZE 时,若队列的开始端空着,又可从头使用空着的空间。当 frontMAXSIZE 时也是一样。

我们采用第2种方法

5️⃣:入队a7,q -> front = 2, q -> rear = 0

解决顺序队列假溢出

引入循环队列

base[0] 接在 base[MAXSIZE -1] 之后,若 rear + 1 == M,则令 rear = 0

实现方法: 利用 模(mod,C语言中: %)运算

插入元素:

// 队尾入队
q -> base[q -> rear] = data;
// 更新队尾指针
q -> rear = (q -> rear + 1) % MAXSIZE;

删除元素:

// 队头元素出队
data = q -> base[q -> front];
// 更新队头指针
q -> front = (q -> front + 1) % MAXSIZE;

循环队列判空、判满冲突

循环队列判满、判空解决方案

解决方案:

1.另外设一个标致以区别队空、队满

2.另设一个变量,记录元素个数

3.少用一个元素空间

本文实现的方法就是第三种。

少用一个元素空间的循环队列判满

循环队列常规操作

/********************* 循环队列的常规操作 **************************/

Queue    InitQueue();             	// 初始化循环队列
int      QueueFull();               // 循环队列判满
int      QueueEmpty();              // 循环队列判空
int      QueueLength();             // 求循环队列长度(元素个数)
int      EnQueue();                 // 循环队列 入队
int      DeQueue();                 // 循环队列 出队

void     QueueStatus();             // 打印队满、队空、队长状态
/****************************************************************/

定义循环队列结构体

#include "stdio.h"
#include "malloc.h"

#define TRUE 1
#define FALSE 0
#define MAXSIZE 10

typedef int ElemType;


// 定义循环队列结构体
typedef struct Queue{ 
   

	int *base;	// 队列首地址
	int front;	// 队列头下标
	int rear;	// 队列尾下标

}*Queue;

初始化循环队列

/* * 初始化循环队列 */
Queue InitQueue(){ 
   
    Queue q;
    // 分配循环队列空间
    q -> base = (ElemType *)malloc(sizeof(ElemType) * MAXSIZE);
    q -> front = q -> rear = 0;
    return q;
}

循环队列判满

/* * 循环队列判满 * q 循环队列 */
int QueueFull(Queue q){ 
   
    if(q == NULL){ 
   
        return FALSE;
    }
    return (q -> rear + 1) % MAXSIZE == q -> front;
}

循环队列判空

/* * 循环队列判空 * q 循环队列 */
int QueueEmpty(Queue q){ 
   
    if(q == NULL){ 
   
        return FALSE;
    }
    return q -> front == q -> rear;
}

计算循环队列的长度

/* * 计算循环队列长度 * q 循环队列 */
int QueueLength(Queue q){ 
   
    if(q == NULL){ 
   
        return FALSE;
    }
    return (q -> rear - q -> front + MAXSIZE) % MAXSIZE;
}

循环队列 入队(EnQueue)

/* * 循环队列 入队 * q 循环队列 * data 入队元素 */
int EnQueue(Queue q, ElemType data){ 
   
    if(QueueFull(q)){ 
   
        return FALSE;
    }
    // 队尾入队
    q -> base[q -> rear] = data;
    // 更新队尾指针
    q -> rear = (q -> rear + 1) % MAXSIZE;
    return TRUE;
}

循环队列 出队(DeQueue)

/* * 循环队列 出队 * q 循环队列 * *val 用来存出队元素的数据 */
int DeQueue(Queue q, ElemType *val){ 
   
    if(QueueEmpty(q)){ 
   
        return FALSE;
    }
    // 队头元素出队
    *val = q -> base[q -> front];
    // 更新队头指针
    q -> front = (q -> front + 1) % MAXSIZE;
    return TRUE;
}

循环队列各操作测试

/* * 打印队满、队空、队长状态 * q 循环队列 */
void QueueStatus(Queue q){ 
   
    printf("QueueFull():%d\n", QueueFull(q));
    printf("QueueEmpty():%d\n", QueueEmpty(q));
    printf("QueueLength():%d\n\n", QueueLength(q));
}

// 程序主入口
int main(int argc, char const *argv[])
{ 
      
    Queue q = InitQueue();
    printf("QueueMaxSize: %d\n\n", MAXSIZE);
    QueueStatus(q); // 打印队满、队空、队长状态

    // 入队
    printf("EnQueue():");
    for(int i = 1; i < MAXSIZE * 2; i+=2){ 
   
        printf("%d\t", i);
        EnQueue(q, i);
    }

    printf("\n");
    QueueStatus(q);

    // 出队
    int val;
    printf("DeQueue():");
    while(!QueueEmpty(q)){ 
   
        DeQueue(q, &val);
        printf("%d\t", val);
    }
    printf("\n");
    QueueStatus(q);

    // 测试循环队列是否会假溢出
    int num = 20;
    printf("EnQueue(%d): %d\t(0 Failed, 1 Success)\n", num, EnQueue(q, num));
    QueueStatus(q);
    return 0;
}

结果如下:

QueueMaxSize: 10

QueueFull():0
QueueEmpty():1
QueueLength():0

EnQueue():1     3       5       7       9       11      13      15      17      19
QueueFull():1
QueueEmpty():0
QueueLength():9

DeQueue():1     3       5       7       9       11      13      15      17
QueueFull():0
QueueEmpty():1
QueueLength():0

EnQueue(20): 1(0 Failed, 1 Success)
QueueFull():0
QueueEmpty():0
QueueLength():1

源代码

源代码已上传到 GitHub Data-Structure-of-C,欢迎大家下载 C语言实现数据结构

✍ 码字不易,记得点赞 ? 收藏 ⭐️

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/138913.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • CompoundButton

    CompoundButtonCompoundButton具有两种状态的按钮,选中和未选中。当按钮被按下或点击时,状态会自动改变。这是一个抽象类,目前有的子类有复选框,单选按钮,开关,切换按钮。 复选框 复选框是一种特定类型的双状态按钮,可以选中或取消选中。 单选按钮 单选按钮是两个状态的按钮,可以选中也可以取消选中。 转变 Switch是一个双态切换开关小部件,可以在两个选项之间进行选择。 …

  • Mysql数据库备份(一)——数据库备份和表备份[通俗易懂]

    一、Mysql中的数据备份:Mysql中数据备份使用的命令是:mysqldump命令将数据库中的数据备份成一个文本文件。表的结构和表中的数据将存储在生成的文本文件中。mysqldump命令的工作原理很简单。它先查出需要备份的表的结构,再在文本文件中生成一个CREATE语句。然后,将表中的所有记录转换成一条INSERT语句。然后通过这些语句,就能够创建表并插入数据。1、Mys

  • myeclipse添加svn插件「建议收藏」

    myeclipse添加svn插件「建议收藏」转载:http://www.cnblogs.com/xdp-gacl/p/3497016.htmlMyEclipse使用总结——MyEclipse10安装SVN插件一、下载SVN插件subclipse下载地址:http://subclipse.tigris.org/servlets/ProjectDocumentList?folderID=2240在打开的网站中

  • JAVA-常用API之StringBuilder

    JAVA-常用API之StringBuilderJAVA-常用API之StringBuilder

  • jbpm工作流 php,jBPM工作流组件

    jbpm工作流 php,jBPM工作流组件jBPM工作流组件如下图所示-1.开始事件它是该过程的起始节点。每个进程只有一个启动节点。此节点仅包含一个没有任何传入连接的传出连接。它具有以下属性:Id:节点的ID,它也应该是独一无二的。Name:节点的名称。2.结束事件它是流程的结束节点。进程可以包含多个End事件。此节点仅包含一个传入连接,不包含传出连接。它具有以下属性:Id:节点的ID,它也应该是独一无二的。Name:节点…

  • 超全,7种经典推荐算法模型及应用

    超全,7种经典推荐算法模型及应用本文调研了推荐系统里的经典推荐算法,结合论文及应用进行分析、归纳并总结成文,既是自己的思考过程,也可当做以后的翻阅手册。前言个性化推荐,是指通过分析、挖掘用户行为,发现用户的个性化需求与兴趣特点,将用户可能感兴趣的信息或商品推荐给用户。本文调研了推荐系统里的经典推荐算法,结合论文及应用进行分析、归纳并总结成文,既是自己的思考过程,也可当做以后的翻阅手册。俗话说学而时习之,人的认识过程是呈螺旋式上升的,特别是理论应用到实践的过程,理论是实践的基础,实践能反过来指导人对理论的认识,我相信在将下文所述的算法应

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号