C语言数据结构(超详细讲解)| 栈和队列的实现

news2025/1/18 10:55:34

栈和队列的实现

引言

在计算机科学的世界里,数据结构是我们构建各种复杂算法和解决问题的基石。其中,栈(Stack)和队列(Queue)作为两种经典的数据结构,在解决实际问题时发挥着不可或缺的作用。

栈和队列的概念或许在我们的日常生活中并不陌生。例如,当你在超市排队结账时,你就是处于一个队列中;而当你使用浏览器的“返回”按钮时,实际上就是在操作一个栈。然而,虽然我们可以轻松地理解这些概念在日常生活中的应用,但是了解它们在计算机科学中的实现和应用,则能为我们打开全新的认知世界。

本篇博客将深入探讨栈和队列的实现原理以及它们在C语言中的实现方式。我们将从基本概念开始,逐步深入,通过代码示例和实际应用场景,带领读者领略这两种数据结构的魅力所在。

让我们一起踏上这段关于数据结构之美的探索之旅吧!

目录

  • 栈和队列的实现
    • 引言
    • 1. 栈
      • 1.1 栈的概念
      • 1.2 栈的实现——数组结构
        • 1.2.1 数组实现栈的结构
        • 1.2.2 栈的初始化
        • 1.2.3 栈的销毁
        • 1.2.4 栈的插入
        • 1.2.5 栈的删除
        • 1.2.6 栈顶元素的获取
        • 1.2.7 栈的判空
        • 1.2.8 获取栈中元素个数
      • 1.3 栈的OJ题
        • 1.3.1 括号匹配问题
    • 2.队列
      • 2.1 队列的概念
      • 2.2 队列的实现——链表结构
        • 2.2.1 链表实现队列结构
        • 2.2.2 队列的初始化
        • 2.2.3 队列的增(尾插)
        • 2.2.4 队列的删(头删)
        • 2.2.5 队列返回对头
        • 2.2.6 队列返回队尾
        • 2.2.7 队列的判空
        • 2.2.8 队列的销毁
        • 2.2.9 队列的大小
      • 2.3 队列的OJ题
        • 2.3.1 设计循环队列

1. 栈

1.1 栈的概念

栈:一种特殊的线性表,其只允许在固定的一端进入插入和删除元素操作。进行数据插入和删除操作的一端叫做栈顶,另一端叫做栈底。 栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。
压栈:栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶
出栈:栈的删除操作叫做出栈,出数据也在栈顶

在这里插入图片描述
示例 栈的压栈和出栈规则(此图TOP指向栈顶元素,我们现实时,使用TOP指向栈顶元素的下一个元素位置)

在这里插入图片描述


1.2 栈的实现——数组结构

栈的实现一般可以使用数组或者链表实现,相对而言数组的结构实现更优一些。因为数组在尾上插入数据的代价比较小。

1.2.1 数组实现栈的结构
typedef int STDateType;

typedef struct Stack
{
	STDataType* a;//指针 指向数组
	int top;	  //栈顶
	int capacity; //容量
}ST;
1.2.2 栈的初始化
void STInit(ST* pst)	//传入一个指向栈结构的指针
{
	//首先先判断给的栈指针是否为NULL
	assert(pst);
	//初始化
	pst->a = NULL;		//将指向数组的指针置为空
	pst->top = 0;		//top指向的是栈顶数据的下一个位置
	//pst->top = -1;      top指向的是栈顶数据
	pst->capacity = 0;
}
1.2.3 栈的销毁
void STDestory(ST* pst)
{
	assert(pst);
	free(pst->a);	//free掉指向数组的指针
	pst->a = NULL;  //并将指针置为空
	pst->capacity = pst->top = 0;	//计数和栈顶标记都置为0
}
1.2.4 栈的插入
void STPush(ST* pst, STDataType x)	//传入一个指向栈结构的指针,和要插入的数据
{
	assert(pst);

	//如果没有空间插入需要申请空间扩容
	if (pst->capacity == pst->top)
	{
		int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;	//调大计数范围
		STDataType* tmp = (STDataType*)realloc(pst->a, newcapacity * sizeof(STDataType));	//运用realloc函数扩大栈里的空间
		//检查是否申请成功
		if (tmp == NULL)
		{
			perror("realloc fail");
			return;
		}
		pst->a = tmp;	//将(数组)空间赋给指向数组的指针
		pst->capacity = newcapacity;	
	}
	pst->a[pst->top] = x;	//插入第一个元素
	pst->top++;	//栈顶++
}
1.2.5 栈的删除
void STPop(ST* pst)
{
	assert(pst);
	assert(pst->top > 0);//要删除数据需先确保有数据

	pst->top--;	//栈顶指针向下移动一个位置,表示弹出一个元素
				//这里的指针并不是真的指针,依然是变量,只是为了更好理解栈顶的变化
				//下面涉及到类似的也是同理
}
1.2.6 栈顶元素的获取
STDataType STTop(ST* pst)	//传入一个指向栈结构的指针,并返回栈顶元素
{
	assert(pst);
	assert(pst->top > 0);	//要获取数据需先确保有数据
	return pst->a[pst->top - 1];	//通过指向数组的指针调取数组,并用栈顶-1获取下标,实现数组元素的访问
}
1.2.7 栈的判空
bool STEmpty(ST* pst)   //判空返回值应为TRUE FALSE
{
	assert(pst);
	if (pst->top == 0)	//如果栈顶指针指向0,说明栈空间没有元素了
	{
		return true;
	}
	return false;
	//return pst->a[pst->top - 1];	//法二:如果有数据就会返回TRUE,没有则返回FALSE
}
1.2.8 获取栈中元素个数
int STSize(ST* pst)
{
	assert(pst);
	return pst->top;
}

代码总览

//Stack.h 文件
#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<assert.h>

//#define 定义宏尽量不要加;
//typedef 重命名需要加;

//用数组来实现栈
typedef int STDataType;

typedef struct Stack
{
	STDataType* a;//指针 指向数组
	int top;	  //栈顶
	int capacity; //容量
}ST;


//栈的初始化
void STInit(ST * pst);

//栈的销毁
void STDestory(ST* pst);

//栈的插入
void STPush(ST* pst,STDataType x);

//栈的删除
void STPop(ST* pst);

//栈顶元素的获取
STDataType STTop(ST* pst);

//栈的判空
bool STEmpty(ST* pst);   //判空返回值应为TRUE FALSE

//获取栈中元素个数
int STSize(ST* pst);

//Stack.c文件
#include"Stack.h"



//栈的初始化
void STInit(ST* pst)
{
	//首先先判断给的栈指针是否为NULL
	assert(pst);
	//初始化
	pst->a = NULL;		//将指向数组的指针置为空
	pst->top = 0;		//top指向的是栈顶数据的下一个位置
	//pst->top = -1;      top指向的是栈顶数据
	pst->capacity = 0;
}

//栈的销毁
void STDestory(ST* pst)
{
	assert(pst);
	free(pst->a);	//free掉指向数组的指针
	pst->a = NULL;  //并将指针置为空
	pst->capacity = pst->top = 0;	//计数置为0
}

//栈的插入
void STPush(ST* pst, STDataType x)
{
	assert(pst);

	//如果没有空间插入需要申请空间扩容
	if (pst->capacity == pst->top)
	{
		int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;	//调大计数范围
		STDataType* tmp = (STDataType*)realloc(pst->a, newcapacity * sizeof(STDataType));	//调大栈里的空间
		//检查是否申请成功
		if (tmp == NULL)
		{
			perror("realloc fail");
			return;
		}
		pst->a = tmp;	//将(数组)空间赋给指向数组的指针
		pst->capacity = newcapacity;	
	}
	pst->a[pst->top] = x;	//插入第一个元素
	pst->top++;	//栈顶++
}

//栈的删除
void STPop(ST* pst)
{
	assert(pst);
	assert(pst->top > 0);//要删除数据需先确保有数据

	pst->top--;
}

//栈顶元素的获取
STDataType STTop(ST* pst)
{
	assert(pst);
	assert(pst->top > 0);
	return pst->a[pst->top - 1];	//通过指向数组的指针调取数组,并用栈顶-1获取下标,实现数组元素的访问
}

//栈的判空
bool STEmpty(ST* pst)   //判空返回值应为TRUE FALSE
{
	assert(pst);
	if (pst->top == 0)
	{
		return true;
	}
	return false;
	//return pst->a[pst->top - 1];	//如果有数据就会返回TRUE,没有则返回FALSE
}
//获取栈中元素个数
int STSize(ST* pst)
{
	assert(pst);
	return pst->top;
}

//tast.c 文件
#include"Stack.h"
//栈的接口
int main()
{
	ST s;
	STInit(&s);
	STPush(&s, 1);
	STPush(&s, 2);
	STPush(&s, 3);
	STPush(&s, 4);

	printf("%d\n",STTop(&s));

	STPop(&s);
	STPop(&s);
	printf("%d\n", STTop(&s));

	STDestory(&s);
   
	return 0;
}

1.3 栈的OJ题

1.3.1 括号匹配问题

在这里插入图片描述

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>

#include"Stack.h"

//用栈先进后出的特性来实现括号匹配的判定
//遇到左括号就入栈,直到遇到右括号,就输出栈顶括号进行匹配
bool isValid(char* s) {
    //初始化栈
    ST st;
    STInit(&st);

    //while循环来读数据
    while (*s)
    {
        //如果读到的数据是左括号就进栈
        if (*s == '('
            || *s == '['
            || *s == '{')
        {
            STPush(&st, *s);
            ++s;    //字符串指针++
        }
        else
        {
            //遇到右括号,但是栈里没有数据,说明不匹配,返回FALSE
            if (STEmpty(&st))
            {
                STDestory(&st);
                return false;
            }
            //如果栈里有数据,就读一个数据对比一下是否不同,然后扔掉配对成功的。
            STDataType top = STTop(&st);
            STPop(&st);
            if ((*s == '}' && top != '{')
                || (*s == ']' && top != '[')
                || (*s == ')' && top != '('))
            {
                STDestory(&st);
                return false;
            }
            else
            {
                ++s;
            }
        }

    }

    //如果栈不是空,说明栈中还有左括号未出
    //没有匹配,返回false
    bool ret = STEmpty(&st);
    STDestory(&st);
    return ret;
}

int main()
{
    char s = '(';
    bool ret = isValid(&s);
    printf("%d",ret);
	return 0;
}

2.队列

2.1 队列的概念

队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出FIFO(First In First Out)的原则。
入队列:进行插入操作的一端称为队尾
出队列:进行删除操作的一端称为队头
在这里插入图片描述

示例 队列的进队和出队规则

在这里插入图片描述


2.2 队列的实现——链表结构

队列也可以使用数组和链表来实现,使用链表的结构实现更优一些,因为如果使用数组结构,出队列在数组头上出数据,效率会比较低。

2.2.1 链表实现队列结构
//定义数据类型和结构体(节点)(双指针+计数)
typedef int QuDataType;

typedef struct QueueNode	//第一个结构体来实现链表节点结构
{
	struct QueueNode* next;
	QuDataType val;
}QNode;

typedef struct QueueP	//第二个结构体来实现两个指针,一个指向链表头,一个指向链表尾
{
	struct QueueNode* phead;
	struct QueueNode* ptail;
	int size;	//记录队列中元素的个数
}Queue;
2.2.2 队列的初始化
void QueueInit(Queue* pq)
{
	assert(pq);
	pq->phead = NULL;
	pq->ptail = NULL;
	pq->size = 0;
}
2.2.3 队列的增(尾插)
void QueuePush(Queue* pq, QuDataType x)
{
	assert(pq);

	//为新节点申请空间
	QNode* newnode = (QNode*)malloc(sizeof(QNode));	//melloc函数只需要传要开辟的空间大小(变量大小而不是指针大小)即可
	//判断节点是否申请成功
	if (newnode == NULL)
	{
		perror("malloc fail");
		return;
	}
	//对新节点进行初始化
	newnode->next = NULL;
	newnode->val = x;

	//如果原链表为空
	if (pq->ptail == NULL)
	{
		pq->phead = pq->ptail = newnode;
	}
	else
	{
		pq->ptail->next = newnode;  //让新节点与原链表建立关系
		pq->ptail = newnode;
	}

	pq->size++;
}

2.2.4 队列的删(头删)
void QueuePop(Queue* pq) 
{
	assert(pq);
	assert(pq->ptail != NULL);

	//如果链表只有一个节点
	if (pq->size == 1)
	{
		free(pq->phead);
		pq->phead = pq->ptail = NULL;
	}
	//如果链表有多个节点
	else
	{
		QNode* Next = pq->phead->next;	//为了free掉phead之后还能找到phead->next
		free(pq->phead);
		pq->phead = Next;
	}

	pq->size--;
}

2.2.5 队列返回对头
QuDataType QueueHead(Queue* pq)
{	
	assert(pq);
	assert(pq->ptail != NULL);
	return pq->phead->val;	//返回队头的值
}
2.2.6 队列返回队尾
QuDataType Queuetail(Queue* pq) 
{
	assert(pq);
	assert(pq->ptail != NULL);
	return pq->ptail->val;	//返回队尾的值

}
2.2.7 队列的判空
bool QueueEmpty(Queue* pq) 
{
	assert(pq);
	return pq->size == 0;	//队列为空返回TRUE 不为空返回FALSE
}
2.2.8 队列的销毁
void QueueDestroy(Queue* pq)
{
	assert(pq);
	QNode* cur = pq->phead;
	while(cur)
	{
		QNode* Next = cur->next;	//为了free掉cur之后还能找到cur->next
		free(cur);
		cur = Next;
	}
	pq->phead = NULL;
	pq->ptail = NULL;
	pq->size = 0;
}
2.2.9 队列的大小
int QueueSize(Queue* pq) 
{
	assert(pq);
	return pq->size;
}

代码总览

//queue.h 文件
#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>

//使用链表实现队列

//定义数据类型和结构体(节点)(双指针+计数)
typedef int QuDataType;

typedef struct QueueNode
{
	struct QueueNode* next;
	QuDataType val;
}QNode;

typedef struct QueueP
{
	struct QueueNode* phead;
	struct QueueNode* ptail;
	int size;	//记录队列中元素的个数
}Queue;

//队列的初始化
void QueueInit(Queue * pq);

//队列的增(尾插)
void QueuePush(Queue* pq, QuDataType x);

//队列的删(头删)
void QueuePop(Queue* pq);

//队列返回对头
QuDataType QueueHead(Queue* pq);

//队列返回队尾
QuDataType Queuetail(Queue* pq);

//队列的判空
bool QueueEmpty(Queue* pq);

//队列的销毁
void QueueDestroy(Queue* pq);

//队列的大小
int QueueSize(Queue* pq);
//queue.c 文件
#include"queue.h"



//队列的初始化
void QueueInit(Queue* pq)
{
	assert(pq);
	pq->phead = NULL;
	pq->ptail = NULL;
	pq->size = 0;

}



//队列的增(尾插)
void QueuePush(Queue* pq, QuDataType x)
{
	assert(pq);

	//为新节点申请空间
	QNode* newnode = (QNode*)malloc(sizeof(QNode));	//melloc函数只需要传要开辟的空间大小(变量大小而不是指针大小)即可
	//判断节点是否申请成功
	if (newnode == NULL)
	{
		perror("malloc fail");
		return;
	}
	//对新节点进行初始化
	newnode->next = NULL;
	newnode->val = x;

	//如果原链表为空
	if (pq->ptail == NULL)
	{
		pq->phead = pq->ptail = newnode;
	}
	else
	{
		pq->ptail->next = newnode;  //让新节点与原链表建立关系
		pq->ptail = newnode;
	}

	pq->size++;
}




//队列的删(头删)
void QueuePop(Queue* pq) 
{
	assert(pq);
	assert(pq->ptail != NULL);

	//如果链表只有一个节点
	if (pq->size == 1)
	{
		free(pq->phead);
		pq->phead = pq->ptail = NULL;
	}
	//如果链表有多个节点
	else
	{
		QNode* Next = pq->phead->next;
		free(pq->phead);
		pq->phead = Next;
	}

	pq->size--;
}




//队列返回队头
QuDataType QueueHead(Queue* pq)
{	
	assert(pq);
	assert(pq->ptail != NULL);
	return pq->phead->val;	//返回队头的值
}


//队列返回队尾
QuDataType Queuetail(Queue* pq) 
{
	assert(pq);
	assert(pq->ptail != NULL);
	return pq->ptail->val;	//返回队尾的值

}


//队列的判空
bool QueueEmpty(Queue* pq) 
{
	assert(pq);
	return pq->size == 0;	//队列为空返回TRUE 不为空返回FALSE
}

//队列的销毁
void QueueDestroy(Queue* pq)
{
	assert(pq);
	QNode* cur = pq->phead;
	while(cur)
	{
		QNode* Next = cur->next;	//为了free掉cur之后还能找到cur->next
		free(cur);
		cur = Next;
	}
	pq->phead = NULL;
	pq->ptail = NULL;
	pq->size = 0;
}

//队列的大小
int QueueSize(Queue* pq) 
{
	assert(pq);
	return pq->size;
}
//test.c 文件
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include"queue.h"

int main()
{
	Queue q;	//定义一个双指针结构体
	QueueInit(&q);
	QueuePush(&q, 1);
	QueuePush(&q, 2);
	printf("%d ", QueueHead(&q));
	QueuePop(&q);

	QueuePush(&q, 3);
	QueuePush(&q, 4);

	while (!QueueEmpty(&q))
	{
		printf("%d ", QueueHead(&q));
		QueuePop(&q);
	}
	printf("\n");


	return 0;
}

环形队列
实际中我们还有一种队列叫循环队列。如操作系统课程讲解生产者消费者模型时,就会使用循环队列。环形队列可以使用数组实现,也可以使用循环列表实现。

在这里插入图片描述
在这里插入图片描述

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

#define MAX_SIZE 10 // 定义环形队列的最大容量

typedef struct {
    int *queue; // 队列数组指针
    int capacity; // 队列容量
    int head; // 队头指针
    int tail; // 队尾指针
    int size; // 队列当前元素个数
} CircularQueue;

// 初始化环形队列
CircularQueue* initQueue(int capacity) {
    CircularQueue *cq = (CircularQueue*)malloc(sizeof(CircularQueue));
    cq->queue = (int*)malloc(capacity * sizeof(int));
    cq->capacity = capacity;
    cq->head = 0;
    cq->tail = 0;
    cq->size = 0;
    return cq;
}

// 入队操作
void enqueue(CircularQueue *cq, int item) {
    if (cq->size == cq->capacity) { // 如果队列已满
        printf("队列已满,无法添加元素\n");
        return;
    }
    cq->queue[cq->tail] = item; // 将元素添加到队尾
    cq->tail = (cq->tail + 1) % cq->capacity; // 更新队尾指针,考虑环形结构
    cq->size++; // 更新队列大小
}

// 出队操作
int dequeue(CircularQueue *cq) {
    if (cq->size == 0) { // 如果队列为空
        printf("队列为空,无法删除元素\n");
        return -1;
    }
    int item = cq->queue[cq->head]; // 获取队头元素
    cq->head = (cq->head + 1) % cq->capacity; // 更新队头指针,考虑环形结构
    cq->size--; // 更新队列大小
    return item;
}

// 判空操作
int isEmpty(CircularQueue *cq) {
    return cq->size == 0;
}

// 判满操作
int isFull(CircularQueue *cq) {
    return cq->size == cq->capacity;
}

// 打印队列元素
void printQueue(CircularQueue *cq) {
    printf("队列元素: ");
    int i = cq->head;
    for (int count = 0; count < cq->size; count++) {
        printf("%d ", cq->queue[i]);
        i = (i + 1) % cq->capacity;
    }
    printf("\n");
}

// 主函数
int main() {
    CircularQueue *cq = initQueue(MAX_SIZE);

    // 示例:入队
    enqueue(cq, 1);
    enqueue(cq, 2);
    enqueue(cq, 3);
    printQueue(cq); // 预期输出: 队列元素: 1 2 3

    // 示例:出队
    int item = dequeue(cq); // 出队一个元素
    printf("出队元素: %d\n", item); // 预期输出: 出队元素: 1
    printQueue(cq); // 预期输出: 队列元素: 2 3

    // 示例:判空和判满
    printf("队列是否为空: %s\n", isEmpty(cq) ? "是" : "否"); // 预期输出: 队列是否为空: 否
    printf("队列是否已满: %s\n", isFull(cq) ? "是" : "否"); // 预期输出: 队列是否已满: 否

    // 示例:继续入队
    enqueue(cq, 4);
    enqueue(cq, 5);
    printQueue(cq); // 预期输出: 队列元素: 2 3 4 5

    free(cq->queue);
    free(cq);
    return 0;
}

在环形队列中,我们使用取模运算来实现队列指针的循环移动。下面我将详细解释一下取模运算在环形队列中的原理:

循环队列的结构:
在普通的线性队列中,队列指针(如队头指针和队尾指针)的范围是从 0 到队列容量减一。而在循环队列中,队尾指针在达到队列容量时不再继续增加,而是从队列的开头重新开始,形成了一个环形结构。

取模运算的作用:
在循环队列中,我们使用取模运算(%)来实现队列指针的循环移动。具体来说,当队尾指针需要向后移动时,我们计算新的队尾位置为 (tail + 1) % capacity,其中 tail 是当前队尾指针的位置,capacity 是队列的容量。这样可以保证当队尾指针达到队列容量时,取模运算会使其重新回到队列的开头,实现了环形结构。

示例:
假设队列容量为 5,当前队尾指针 tail 为 4,即指向队列的最后一个位置。如果此时需要再入队一个元素,按照普通的线性队列逻辑,下一个位置应该是 tail + 1,即 5,但这个位置超出了队列的范围。而在环形队列中,我们使用取模运算:5 % 5 = 0,因此新的队尾指针位置为 0,指向队列的开头,实现了循环移动。

取模运算的这种特性使得循环队列的操作更加高效,并且不需要额外的移动操作来处理队列指针达到边界的情况,因为取模运算会自动将其映射回队列的有效范围内。


2.3 队列的OJ题

2.3.1 设计循环队列

在这里插入图片描述

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

// 定义循环队列的结构体
typedef struct {
    int *array;     // 队列数组指针
    int capacity;   // 队列容量
    int size;       // 队列当前元素个数
    int front;      // 队头指针
    int rear;       // 队尾指针
} MyCircularQueue;

// 创建一个新的循环队列
MyCircularQueue* myCircularQueueCreate(int k) {
    // 分配循环队列结构体的内存空间
    MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));
    // 分配队列数组的内存空间
    obj->array = (int*)malloc(k * sizeof(int));
    obj->capacity = k;  // 设置队列容量
    obj->size = 0;      // 初始化队列当前元素个数为 0
    obj->front = 0;     // 初始化队头指针为 0
    obj->rear = -1;     // 初始化队尾指针为 -1
    return obj;
}

// 向循环队列中插入一个元素
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {
    // 如果队列已满,则返回 false
    if (myCircularQueueIsFull(obj)) {
        return false;
    }
    // 队尾指针后移一位,考虑到环形结构,使用取模运算确保指针位置在合法范围内
    obj->rear = (obj->rear + 1) % obj->capacity;
    // 将元素值插入到队尾指针所指向的位置
    obj->array[obj->rear] = value;
    obj->size++;  // 更新队列的当前元素个数
    return true;
}

// 从循环队列中删除一个元素
bool myCircularQueueDeQueue(MyCircularQueue* obj) {
    // 如果队列为空,则返回 false
    if (myCircularQueueIsEmpty(obj)) {
        return false;
    }
    // 队头指针后移一位,考虑到环形结构,使用取模运算确保指针位置在合法范围内
    obj->front = (obj->front + 1) % obj->capacity;
    obj->size--;  // 更新队列的当前元素个数
    return true;
}

// 获取循环队列的队首元素
int myCircularQueueFront(MyCircularQueue* obj) {
    // 如果队列为空,则返回 -1
    return myCircularQueueIsEmpty(obj) ? -1 : obj->array[obj->front];
}

// 获取循环队列的队尾元素
int myCircularQueueRear(MyCircularQueue* obj) {
    // 如果队列为空,则返回 -1
    return myCircularQueueIsEmpty(obj) ? -1 : obj->array[obj->rear];
}

// 检查循环队列是否为空
bool myCircularQueueIsEmpty(MyCircularQueue* obj) {
    return obj->size == 0;  // 如果队列当前元素个数为 0,则为空
}

// 检查循环队列是否已满
bool myCircularQueueIsFull(MyCircularQueue* obj) {
    return obj->size == obj->capacity;  // 如果队列当前元素个数等于容量,则已满
}

// 释放循环队列的内存空间
void myCircularQueueFree(MyCircularQueue* obj) {
    free(obj->array);
    free(obj);
}

int main() {
    // 创建一个容量为 3 的循环队列
    MyCircularQueue* circularQueue = myCircularQueueCreate(3);
    printf("%d\n", myCircularQueueEnQueue(circularQueue, 1));  // 返回 true
    printf("%d\n", myCircularQueueEnQueue(circularQueue, 2));  // 返回 true
    printf("%d\n", myCircularQueueEnQueue(circularQueue, 3));  // 返回 true
    printf("%d\n", myCircularQueueEnQueue(circularQueue, 4));  // 返回 false,队列已满
    printf("%d\n", myCircularQueueRear(circularQueue));  // 返回 3
    printf("%d\n", myCircularQueueIsFull(circularQueue));  // 返回 true
    printf("%d\n", myCircularQueueDeQueue(circularQueue));  // 返回 true
    printf("%d\n", myCircularQueueEnQueue(circularQueue, 4));  // 返回 true
    printf("%d\n", myCircularQueueRear(circularQueue));  // 返回 4
    myCircularQueueFree(circularQueue);
    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【回溯 代数系统】679. 24 点游戏

本文涉及知识点 回溯 代数系统 LeetCode679. 24 点游戏 给定一个长度为4的整数数组 cards 。你有 4 张卡片&#xff0c;每张卡片上都包含一个范围在 [1,9] 的数字。您应该使用运算符 [‘’, ‘-’, ‘*’, ‘/’] 和括号 ‘(’ 和 ‘)’ 将这些卡片上的数字排列成数学表达式…

【C#】WPF加载浏览器

结果展示 下载SDK 前端代码 红色框住的为添加代码 <Window x:Class"WPFwebview.MainWindow"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d"http://…

TCP四次挥手——断开连接 滑动窗口-流量控制

四次挥手 在TCP的四次挥手中&#xff0c;其重要作用就是释放客户端和服务器的连接。 这里的一些参数非常重要&#xff0c;因为这些参数的作用是为了表达TCP四次挥手断开连接的过程。 其中的参数如下 1.FIN&#xff1a;FIN (Finish) 是TCP协议中的一个标志位&#xff0c;用于…

推荐网站(9)pixabay免费可商用的图片、视频、插画、矢量图、音乐

今天推荐一款可以免费可商用的图片、视频、插画、矢量图、音乐的资源网站&#xff0c;这里面的所以东西都是免费的&#xff0c;并且可以商用。对那些做视频剪辑的人来说帮助非常大。它里面的资源非常的丰富&#xff0c;质量也高。 比如搜索下雨 链接直达&#xff1a;https://pi…

解决webstorm没有vue语法提示;webstorm没有代码提示

解决webstorm没有vue语法提示&#xff1b;webstorm没有代码提示 使用webstorm 2023.x 开发vue项目。发现死活没有vue语法提示&#xff0c;即便是npm install、清理缓存。对比其他vue项目却有语法提示&#xff0c;最后发现依赖库被忽略了&#xff1a; 删除掉node_modules 的忽略…

huggingface:利用git克隆目标资源

前言 因为有很多模型资源都被放在了huggingface上&#xff0c;为了下载它们&#xff0c;着实让一个不懂git的人犯了难&#xff0c;绕了很多远路&#xff0c;甚至将不需要解决的问题也都拿上了台面&#xff0c;因此我将在本篇博客中记载一些关于【huggingface】中利用git克隆目标…

apisix3.9.1 和 dashboard 离线安装

服务器配置 centos7 linux x86 64 前置 需要将离线安装包上传到服务器上 {上传目录 /root/apisix-soft/ } 【建议:优先上传etcd-*.jar \ apisix-*.rpm \ cyrus-*.rpm \ openldap-*.rpm 等安装好apisix后再上传apisix-dashboard-*.rpm】 可以自行网上寻找&#xff0c;或找一台可…

Python接口自动化测试之动态数据处理

在前面的知识基础上介绍了在接口自动化测试中&#xff0c;如何把数据分离出来&#xff0c;并且找到它的共同点&#xff0c;然后依据这个共同点来找到解决复杂问题的思想。我一直认为&#xff0c;程序是人设计的&#xff0c;它得符合人性&#xff0c;那么自动化测试的&#xff0…

使用命令查看电脑最大支持物理内存容量

一、Windows 系统 以管理员身份运行 cmd C:\WINDOWS\system32>wmic memphysical get maxcapacity /format:value将返回值除以1024&#xff0c;再除以1024&#xff0c;即本机最大支持的内存容量 或一行命令直接获取返回值 for /f "tokens2 delims" %I in (wmic…

怎么给视频加水印?2招轻松搞定

在数字媒体时代&#xff0c;视频水印作为一种有效的版权保护手段&#xff0c;被广泛应用于各种场景。给视频添加水印不仅可以防止内容被恶意盗用&#xff0c;还能增加视频的辨识度&#xff0c;提升品牌形象。本文将为您介绍2种简单易行的方法&#xff0c;教您怎么给视频加水印&…

保研机试之【动态规划】

本文为博客&#xff1a;动态规划解题套路框架 | labuladong 的算法笔记 的笔记 前言 动态规划问题的一般形式就是求最值&#xff0c;求解动态规划的核心问题是穷举。动态规划三要素为&#xff1a;最优子结构、重叠子问题、状态转移方程。首先要判断&#xff0c;该问题是否具有…

模电·场效应管放大电路的动态分析_039

场效应管放大电路的动态分析 一、场效应管的低频小信号等效模型二、基本共源放大电路的动态分析三、基本共漏放大电路的动态分析 一、场效应管的低频小信号等效模型 与分析晶体管的 h h h参数等效模型相同&#xff0c;将场效应管也看成一个两端口网络&#xff0c;栅极与源极之间…

MobileNet 网络详解

一、了解 网络亮点&#xff1a; 1、DW网络&#xff0c;大大减少运算量和参数数量 2、增加超参数&#xff1a;控制卷积层卷积核个数的超参数 &#xff0c;控制图像输入大小的超参数 &#xff0c;这两个超参数是人为设定的&#xff0c;不是机器学习到的。 二、DW卷积&#xff…

怎么防止源代码防泄密

随着数字化时代的到来&#xff0c;源代码作为企业和个人的重要资产之一&#xff0c;承载着无价的知识和创新。然而&#xff0c;源代码泄露已成为当今信息安全领域的重要挑战之一&#xff0c;给企业带来了严重的经济损失和声誉风险。面对这一挑战&#xff0c;我们有责任加强对源…

亿级流量系统架构设计与实战

&#x1f482; 个人网站:【 摸鱼游戏】【神级代码资源网站】【工具大全】&#x1f91f; 一站式轻松构建小程序、Web网站、移动应用&#xff1a;&#x1f449;注册地址&#x1f91f; 基于Web端打造的&#xff1a;&#x1f449;轻量化工具创作平台&#x1f485; 想寻找共同学习交…

【火热征稿~~】2024年心理、哲学与历史国际会议(ICPPH 2024)

2024年心理、哲学与历史国际会议&#xff08;ICPPH 2024&#xff09; 2024 International Conference on Psychology, Philosophy, and History 【会议简介】 2024年心理、哲学与历史国际会议将于历史文化名城武汉召开。此次盛会集结了来自世界各地的心理学家、哲学家和历史学…

01软件下载安装和P解

凯哥英语视频 软件下载安装和P解 凯哥英语视频1.官网直接下&#xff0c;专业版安装不会有人不会吧实在下载不到就去我这百度云吧结语 1.官网直接下&#xff0c;专业版 点击前往逛网下载https://www.jetbrains.com/pycharm/ 下载专业版&#xff0c;奶茶外卖都能点&#xff0c;只…

【爬虫】爬取股票数据写入数据库并显示(四)

本文所涉及代码已全部打包上传&#xff0c;需要可以到文章末尾查看获取方式&#xff0c;免费&#xff0c;仅做学习交流&#xff01;&#xff01;&#xff01; 股票客户端软件 2024/05&#xff0c;本文主要内容如下&#xff1a; 使用python requests爬取东方财富官网数据。将爬…

ros1版本apollo感知融合算法

apollo.ros-7.0.0 本文章针对apollo 7.0.0版本进行了ros1移植&#xff0c;具体介绍见下文。 PS&#xff1a;项目开发详细了解可在评论区留言。 ros版本apollo7.0.0感知算法 基于ros1的apollo 7.0.0感知融合算法。 简介 此项目基于ros1的apollo 7.0.0感知融合算法&#xff0…

05. 【Java教程】第一个 Java 程序

本节我们将以Windows操作系统为例&#xff0c;编写并执行第一个Java程序。在这之前&#xff0c;请确保你的操作系统上已经安装了JDK 1. 编译程序 大家可能有个疑问&#xff0c;为什么需要编译程序呢&#xff1f;计算机不能直接执行我们编写的源代码吗&#xff1f; 这是由于计…