【数据结构初阶】第五篇——栈和队列

简介: 【数据结构初阶】第五篇——栈和队列


栈的概念及结构


:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据的插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的元素遵循后进先出的原则。

压栈:栈的插入操作叫做进栈/入栈/压栈,入数据在栈顶。

出栈:栈的删除操作叫做出栈,出数据也在栈顶。

image.png

image.png

栈的实现


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

image.png

结构如下:

typedef int STDataType;//栈中存储的元素类型(这里用整型举例)
typedef struct Stack
{
  STDataType* a;//栈
  int top;//栈顶
  int capacity;//容量,方便增容
}Stack;

栈的初始化


首先,我们需要用结构体创建一个栈,这个结构体需要包括栈的基本内容(栈,栈顶,栈的容量)。

//初始化栈
void StackInit(Stack* pst)
{
  assert(pst);
  pst->a = (STDataType*)malloc(sizeof(STDataType)* 4);//初始化栈可存储4个元素
  pst->top = 0;//初始时栈中无元素,栈顶为0
  pst->capacity = 4;//容量为4
}

销毁栈


因为栈的内存空间是动态开辟出来的,当我们使用完后必须释放其内存空间,避免内存泄漏

//销毁栈
void StackDestroy(Stack* pst)
{
  assert(pst);
  free(pst->a);//释放栈
  pst->a = NULL;//及时置空
  pst->top = 0;//栈顶置0
  pst->capacity = 0;//容量置0
}

入栈


进行入栈操作前,我们需要检测栈的当前状态,若已满,则需要先对其进行增容,然后才能进行入栈操作。

//入栈
void StackPush(Stack* pst, STDataType x)
{
  assert(pst);
  if (pst->top == pst->capacity)//栈已满,需扩容
  {
    STDataType* tmp = (STDataType*)realloc(pst->a, sizeof(STDataType)*pst->capacity * 2);
    if (tmp == NULL)
    {
      printf("realloc fail\n");
      exit(-1);
    }
    pst->a = tmp;
    pst->capacity *= 2;//栈容量扩大为原来的两倍
  }
  pst->a[pst->top] = x;//栈顶位置存放元素x
  pst->top++;//栈顶上移
}

出栈


出栈操作比较简单,即让栈顶的位置向下移动一位即可。但需检测栈是否为空,若为空,则不能进行出栈操作。

//出栈
void StackPop(Stack* pst)
{
  assert(pst);
  assert(!StackEmpty(pst));//检测栈是否为空
  pst->top--;//栈顶下移
}

获取栈顶元素


获取栈顶元素,即获取栈的最上方的元素。若栈为空,则不能获取。

//获取栈顶元素
STDataType StackTop(Stack* pst)
{
  assert(pst);
  assert(!StackEmpty(pst));//检测栈是否为空
  return pst->a[pst->top - 1];//返回栈顶元素
}

检测栈是否为空


检测栈是否为空,即判断栈顶的位置是否是0即可。若栈顶是0,则栈为空。

//检测栈是否为空
bool StackEmpty(Stack* pst)
{
  assert(pst);
  return pst->top == 0;
}

获取栈中有效元素个数


因为top记录的是栈顶,使用top的值便代表栈中有效元素的个数。

//获取栈中有效元素个数
int StackSize(Stack* pst)
{
  assert(pst);
  return pst->top;//top的值便是栈中有效元素的个数
}

队列


队列的概念和结构


队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出FIFO入队列:进行插入操作的一端称为队尾,出队列:进行删除操作的一端称为对头。

image.png

队列的结构,我们选取单链表来实现,秩序进行头删和为插的不足即可。如果选数组,那么每一次删头我们都要挪动一遍数据,这种方式不优,所以我们还是选取用单链表来实现。

定义的结构如下

typedef int QDataType;//队列中存储的元素类型(这里用整型举例)
typedef struct QListNode
{
  struct QListNode* next;//指针域
  QDataType data;//数据域
}QListNode;
typedef struct Queue
{
  QListNode* head;//队头
  QListNode* tail;//队尾
}Queue;

队列的实现


队列的初始化


//初始化队列
void QueueInit(Queue* pq)
{
  assert(pq);
  //起始时队列为空
  pq->head = NULL;
  pq->tail = NULL;
}

销毁队列


队列中的每一个结点所占用的内存空间都是动态开辟的,当我们使用完队列后需要及时释放队列中的每一个结点。

//销毁队列
void QueueDestroy(Queue* pq)
{
  assert(pq);
  QListNode* cur = pq->head;//接收队头
  //遍历链表,逐个释放结点
  while (cur)
  {
    QListNode* next = cur->next;
    free(cur);
    cur = next;
  }
  pq->head = NULL;//队头置空
  pq->tail = NULL;//队尾置空
}

入队


入队列,即申请一个新结点并将其链接到队尾,然后改变队尾的指针指向即可。需要注意的是:若队列中原本无数据,那么我们只需让队头和队尾均指向这个新申请的结点即可。

//队尾入队列
void QueuePush(Queue* pq, QDataType x)
{
  assert(pq);
  QListNode* newnode = (QListNode*)malloc(sizeof(QListNode));//申请新结点
  if (newnode == NULL)
  {
    printf("malloc fail\n");
    exit(-1);
  }
  newnode->data = x;//新结点赋值
  newnode->next = NULL;//新结点指针域置空
  if (pq->head == NULL)//队列中原本无结点
  {
    pq->head = pq->tail = newnode;//队头、队尾直接指向新结点
  }
  else//队列中原本有结点
  {
    pq->tail->next = newnode;//最后一个结点指向新结点
    pq->tail = newnode;//改变队尾指针指向
  }
}

出队


出队列,即释放队头指针指向的结点并改变队头指针的指向即可。若队列中只有一个结点,那么直接将该结点释放,然后将队头和队尾置空即可。

//队头出队列
void QueuePop(Queue* pq)
{
  assert(pq);
  assert(!QueueEmpty(pq));//检测队列是否为空
  if (pq->head->next == NULL)//队列中只有一个结点
  {
    free(pq->head);
    pq->head = NULL;
    pq->tail = NULL;
  }
  else//队列中有多个结点
  {
    QListNode* next = pq->head->next;
    free(pq->head);
    pq->head = next;//改变队头指针指向
  }
}

获取对头元素


获取队列头部元素,即返回队头指针指向的数据即可。

//获取队列头部元素
QDataType QueueFront(Queue* pq)
{
  assert(pq);
  assert(!QueueEmpty(pq));//检测队列是否为空
  return pq->head->data;//返回队头指针指向的数据
}

获取队尾元素


获取队列尾部元素,即返回队尾指针指向的数据即可。

//获取队列尾部元素
QDataType QueueBack(Queue* pq)
{
  assert(pq);
  assert(!QueueEmpty(pq));//检测队列是否为空
  return pq->tail->data;//返回队尾指针指向的数据
}

判断队列是否为空


检测队列是否为空,即判断队头指针指向的内容是否为空。

//检测队列是否为空
bool QueueEmpty(Queue* pq)
{
  assert(pq);
  return pq->head == NULL;
}

获取队列中元素个数


队列中有效元素个数,即队列中的结点个数。我们只需遍历队列,统计队列中的结点数并返回即可。

//获取队列中有效元素个数
int QueueSize(Queue* pq)
{
  assert(pq);
  QListNode* cur = pq->head;//接收队头
  int count = 0;//记录结点个数
  while (cur)//遍历队列
  {
    count++;
    cur = cur->next;
  }
  return count;//返回队列中的结点数
}
目录
打赏
0
0
0
0
15
分享
相关文章
232.用栈实现队列,225. 用队列实现栈
在232题中,通过两个栈(`stIn`和`stOut`)模拟队列的先入先出(FIFO)行为。`push`操作将元素压入`stIn`,`pop`和`peek`操作则通过将`stIn`的元素转移到`stOut`来实现队列的顺序访问。 225题则是利用单个队列(`que`)模拟栈的后入先出(LIFO)特性。通过多次调整队列头部元素的位置,确保弹出顺序符合栈的要求。`top`操作直接返回队列尾部元素,`empty`判断队列是否为空。 两题均仅使用基础数据结构操作,展示了栈与队列之间的转换逻辑。
栈区的非法访问导致的死循环(x64)
这段内容主要分析了一段C语言代码在VS2022中形成死循环的原因,涉及栈区内存布局和数组越界问题。代码中`arr[15]`越界访问,修改了变量`i`的值,导致`for`循环条件始终为真,形成死循环。原因是VS2022栈区从低地址到高地址分配内存,`arr`数组与`i`相邻,`arr[15]`恰好覆盖`i`的地址。而在VS2019中,栈区先分配高地址再分配低地址,因此相同代码表现不同。这说明编译器对栈区内存分配顺序的实现差异会导致程序行为不一致,需避免数组越界以确保代码健壮性。
21 0
栈区的非法访问导致的死循环(x64)
java实现队列数据结构代码详解
本文详细解析了Java中队列数据结构的实现,包括队列的基本概念、应用场景及代码实现。队列是一种遵循“先进先出”原则的线性结构,支持在队尾插入和队头删除操作。文章介绍了顺序队列与链式队列,并重点分析了循环队列的实现方式以解决溢出问题。通过具体代码示例(如`enqueue`入队和`dequeue`出队),展示了队列的操作逻辑,帮助读者深入理解其工作机制。
107 1
|
5月前
|
STL——栈和队列和优先队列
通过以上对栈、队列和优先队列的详细解释和示例,希望能帮助读者更好地理解和应用这些重要的数据结构。
77 11
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
6月前
|
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
299 77
|
6月前
|
C++
【C++数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】
【数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】(1)遇到左括号:进栈Push()(2)遇到右括号:若栈顶元素为左括号,则出栈Pop();否则返回false。(3)当遍历表达式结束,且栈为空时,则返回true,否则返回false。本关任务:编写一个程序利用栈判断左、右圆括号是否配对。为了完成本关任务,你需要掌握:栈对括号的处理。(1)遇到左括号:进栈Push()开始你的任务吧,祝你成功!测试输入:(()))
158 7
|
6月前
|
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
217 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
GitHub 配置 SSH 密钥(详细流程)
GitHub 配置 SSH 密钥(详细流程)
909 0
【一步步开发AI运动APP】七、自定义姿态动作识别检测——之规则配置检测
本文介绍了如何通过【一步步开发AI运动APP】系列博文,利用自定义姿态识别检测技术开发高性能的AI运动应用。核心内容包括:1) 自定义姿态识别检测,满足人像入镜、动作开始/停止等需求;2) Pose-Calc引擎详解,支持角度匹配、逻辑运算等多种人体分析规则;3) 姿态检测规则编写与执行方法;4) 完整示例展示左右手平举姿态检测。通过这些技术,开发者可轻松实现定制化运动分析功能。
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问