[一篇读懂]C语言十二讲:栈与队列和真题实战

news2025/1/23 6:10:18

[一篇读懂]C语言十二讲:栈与队列和真题实战

  • 1. 与408关联解析及本节内容介绍
    • 1 与408关联解析
    • 2 本节内容介绍
  • 2. 栈(stack)的原理解析
    • 2.1 **栈:只允许在一端进行插入或删除操作的线性表**
    • 2.2 栈的基本操作
    • 2.3 栈的顺序存储
    • 2.4 栈的链表存储
  • 3. 初始化栈 - 入栈 - 出栈实战
  • 4. 队列(Queue) - 循环队列原理解析
    • 4.1 **队列:只允许在一端进行插入,而在另一端进行删除的线性表**
    • 4.2 队列的基本操作
    • 4.3 循环队列
      • 循环数列元素入队
      • 循环数列元素出队
    • 4.4 队列的链式存储
      • 存储结构
  • 5. 循环队列实战
  • 6. 队列实战(通过链表实现)
  • 总结
    • 2
    • 3
    • 4
    • 5
    • 6


1. 与408关联解析及本节内容介绍

1 与408关联解析

【2019年队列】
42.(10分)请设计一个队列,要求满足:①初始时队列为空;②入队时,允许增加队列占用空间;③出队后,出队元素所占用的空间可重复使用,即整个队列所占用的空间只增不减;④入队操作和出队操作的时间复杂度始终保持为O(1)。请回答下列问题:
(1)该队列是应选择链式存储结构,还是应选择顺序存储结构?
(2)画出队列的初始状态,并给出判断队空和队满的条件。
(3)画出第一个元素入队后的队列状态。
(4)给出入队操作和出队操作的基本过程。

栈 - 一般为选择题,大题较少;
队列 - 选择题,大题均涉及。

2 本节内容介绍

本节分为六小节讲解,包含栈,队列,循环队列的原理讲解及实战
第一小节是针对栈的原理讲解
第二小节是针对初始化栈-入栈-出栈实战
第三小节是队列-循环队列原理解析
第四小节是循环队列实战
第五小节是队列的实战(通过链表的头插,头部删除实现)
第六小节是2019年42题真题讲解


2. 栈(stack)的原理解析

2.1 栈:只允许在一端进行插入或删除操作的线性表

堆栈(stack)又称为堆叠
栈的特点是:先进后出(First In Last Out)
FILO

2.2 栈的基本操作

(类似电梯)

  • 元素入栈 - 从栈顶(Top)添加一个元素
    元素入栈

  • 元素出栈 - 从栈顶(Top)删除一个元素
    元素出栈

2.3 栈的顺序存储

栈是结构体
用数组存储数据
top是栈顶
顺序存储实现栈

  • 元素入栈&出栈代码实现
//初始化
S.top = -1;//栈为空
//入栈
S.top = S.top + 1;
S.data[S.top] = 1;

S.top = S.top + 1;
S.data[S.top] = 2;

S.top = S.top + 1;
S.data[S.top] = 3;

//可以写成以下形式
//前加加,先做加1,然后再去做后续的
S.data[++S.top] = 4;

//出栈
x = S.data[S.top];
S.top = S.top - 1;

//可以写成以下形式
//后减减
x = S.data[S.top--]
  • 栈满
    S.top等于MaxSize - 1时,栈满

2.4 栈的链表存储

(相对不重要,考的概率极低)

头部插入法和头部删除法
LiStack Lhead = (LiStack)malloc(sizeof(struct Linknode))
Lhead->next = NULL;
在这里插入图片描述>top = (LiStack)malloc(sizeof(struct Linknode));
top->next = NULL;
top->data = 1;
top->next = Lhead->next;
Lhead->next = top;
 
元素出栈
在这里插入图片描述c = top->data;
Lhead->next = top->next;
free(top);
top = Lhead->next;
 
栈空栈满
在这里插入图片描述
Lhead->next == NULL为真,则栈为空
只要存在剩余内存,那么栈就可以继续添加元素


3. 初始化栈 - 入栈 - 出栈实战

代码实战步骤(五步):
初始化栈 - 判断栈是否为空 - 压栈 - 获取栈顶元素 - 弹栈。

注意:S.top为-1时,代表栈为空,每次先对S.top加1后,再放置元素

五个步骤分为五个子函数。
代码演示:

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

#define MaxSize 50
typedef int ElemType;
typedef struct {
	ElemType data[MaxSize];//数组
	int top;//始终指向栈顶的一个变量
}SqStack;

//初始化栈
void InitStack(SqStack& S)
{
	S.top = -1;//初始化栈,让栈为空

}

//判断栈是否为空
bool StackEmpty(SqStack S)
{
	if (-1 == S.top)
	{
		return true;
	}
	else
	{
		return false;
	}
}

//入栈
bool Push(SqStack &S, ElemType x)
{
	//判断栈是否满了
	if (S.top == MaxSize - 1)
	{
		return false;
	}
	S.data[++S.top] = x;//等价于S.top = S.top + 1;  S.data[S.top] = x;
	return true;
}

//获取栈元素
bool GetTop(SqStack S, ElemType& m)
{
	//判断栈是否为空
	if (StackEmpty(S))
	{
		return false;
	}
	m = S.data[S.top];//拿栈顶元素
	return true;
}

//弹出栈元素
bool Pop(SqStack& S, ElemType& m)
{
	//判断栈是否为空
	if (StackEmpty(S))
	{
		return false;
	}
	m = S.data[S.top--];//出栈 - 先m = S.data[S.top];  S.top = S.top - 1
	return true;
}

int main()
{
	SqStack S;
	InitStack(S);
	bool flag;
	flag = StackEmpty(S);
	if (flag)
	{
		printf("stack is empty\n");
	}
	Push(S, 3);//入栈元素3
	Push(S, 4);//入栈元素4
	Push(S, 5);//入栈元素5
	ElemType m;
	flag = GetTop(S, m);//获取栈顶元素
	if (flag)
	{
		printf("获取栈顶元素为 %d\n", m);
	}
	flag = Pop(S, m);//弹出栈顶元素
	if (flag)
	{
		printf("弹出栈顶元素为 %d\n", m);
	}
	flag = GetTop(S, m);//获取栈顶元素
	if (flag)
	{
		printf("获取栈顶元素为 %d\n", m);
	}
	return 0;
}

运行结果:在这里插入图片描述


4. 队列(Queue) - 循环队列原理解析

4.1 队列:只允许在一端进行插入,而在另一端进行删除的线性表

队列(queue)简称
也是一种操作受限
队的特点是:先进先出(First In First Out)
FIFO

4.2 队列的基本操作

队头(Front)。允许删除的一端,又称队首。
队尾(Rear)。允许插入的一端。
在这里插入图片描述

4.3 循环队列

#define MaxSize 6
typedef int ElemType;
typedef struct{
	ElemType data[MaxSize];//数组,存储MaxSize - 1个元素
	int front,rear;//队列头,队列尾
}SqQueue;

SqQueue Q;

循环队列

  1. Q,front和Q,rear初始化为零,其相等时,循环队列为空。
  2. 放入一个元素后,Q.rear+1;
    出队一个元素后,Q.front+1。
  3. 如果全部放满,Q.front和Q.rear相等,无法判断队满还是队空。
  4. (Q.rear + 1) % MaxSize == Q.front时,判断队满,牺牲一个存储单元

循环数列元素入队

bool Enqueue(SqQueue &Q, ElemType x)
{
	if ((Q.rear + 1) % MaxSize == Q.front)
		return false;
	Q.data[Q.rear] = x;//放入元素
	Q.rear = (Q.rear + 1) % MaxSize;//改变队尾标记
	return true;
}

循环数列元素出队

bool Enqueue(DeQueue &Q, ElemType x)
{
	if (Q.rear == Q.front)
		return false;
	x = Q.data[Q.front];//先进先出
	Q.front = (Q.front + 1) % MaxSize;
	return true;/
}

4.4 队列的链式存储

队列的链式表示称为链队列,它实际上是一个同时带有队头指针和队尾指针单链表。头指针指向队头结点,尾指针指向队尾结点,即单链表的最后一个结点。
在这里插入图片描述

存储结构

typedef int ElemType;
typedef struct LinkNode{
	ElemType data;
	struct LinkNode *next;
}LinkNode;  //链表结点的结构体
typedef struct {
	LinkNode *front, *rear; //链表头 链表尾
}LinkQueue; //先进先出

LinkQueue Q;

相对于原有编写的链表增加了尾指针


5. 循环队列实战

代码实战步骤(四步):
初始化循环队列 - 判断循环队列是否为空 - 入队 - 出队

四个步骤分为四个子函数。
代码演示:

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


#define MaxSize 5
typedef int ElemType;
typedef struct {
	ElemType data[MaxSize];//数组,存储MaxSize - 1个元素
	int front, rear;//队列头 队列尾
}SqQueue;

void InitQueue(SqQueue& Q)
{
	Q.front = Q.rear = 0;//初始化循环队列,就是让头和尾都指向零号
}

//判断循环队列是否为空
bool IsEmpty(SqQueue Q)
{
	return Q.rear == Q.front;//相等时为空
}

//入队
bool EnQueue(SqQueue& Q, ElemType x)
{
	if ((Q.rear + 1) % MaxSize == Q.front)//判断是否队满 - 满了就不能入队了
	{
		return false;
	}
	Q.data[Q.rear] = x;//放入元素3 4 5 6
	Q.rear = (Q.rear + 1) % MaxSize;//rear要加1,如果大于数组最大下标,要回到开头
	return true;
}

//出队
bool DeQueue(SqQueue& Q, ElemType& x)
{
	if (Q.rear == Q.front)//队列为空,无法出队
	{
		return false;
	}
	x = Q.data[Q.front];//出队
	Q.front = (Q.front + 1) % MaxSize;
	return true;
}

//循环队列的代码实战
int main()
{
	SqQueue Q;
	bool ret;//存储返回值
	ElemType element;//存储出队元素
	InitQueue(Q);
	ret = IsEmpty(Q);
	if (ret)
	{
		printf("SqQueue is empty\n");
	}
	else
	{
		printf("SqQueue is not empty\n");
	}
	EnQueue(Q, 3);//入队3
	EnQueue(Q, 4);
	EnQueue(Q, 5);
	ret = EnQueue(Q, 6);
	ret = EnQueue(Q, 7);//队满,7无法入队
	if (ret)
	{
		printf("EnQueue successful\n");
	}
	else
	{
		printf("EnQueue failed\n");
	}
	ret = DeQueue(Q, element);//出队
	if (ret)
	{
		printf("DeQueue successful,element = %d\n",element);
	}
	else
	{
		printf("DeQueue failed\n");
	}
	ret = DeQueue(Q, element);//出队
	if (ret)
	{
		printf("DeQueue successful,element = %d\n", element);
	}
	else
	{
		printf("DeQueue failed\n");
	}
	ret = EnQueue(Q, 8);//前面出队了元素,8可以入队
	if (ret)
	{
		printf("EnQueue successful\n");
	}
	else
	{
		printf("EnQueue failed\n");
	}
	return 0;
}

运行结果:在这里插入图片描述


6. 队列实战(通过链表实现)

代码实战步骤(三步):
初始化队列 - 入队 - 出队


总结

2

  • 头插法新建链表流程图:
    1

3

  • 尾插法新建链表流程图:
    1

4

  • 按位置查找流程图:1

5

  • 往第i个位置插入元素流程图:
    1

6

  • 链表中每一个结点在内存中是不连续的,通过单步调试,在变量窗口,依次点开头指针L,观察每一个结点是否符合自己的预期。

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

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

相关文章

PE文件+UPX壳 ida分析

die查壳发现是UPX壳&#xff0c;直接用ida分析&#xff0c;会发现能分析出来的信息特别少&#xff0c;需要脱壳 工具链接发布 UPX/UPX (github.com) 下载压缩包后解压&#xff0c;直接在该文件路径下cmd&#xff0c;输入upx.exe -h安装完成&#xff0c;使用命令“upx -d 文件路…

计算机网络-网络层与链路层协议分析实验

一.实验目的 通过本实验&#xff0c;进一步熟悉PacketTracer的使用&#xff0c;学习路由器与交换机的基本配置&#xff0c;加深对网络层与链路层协议的理解。 二.实验内容 1.完成路由器交换机的基本配置 2.了解 ICMP 数据包的格式 3.检查ARP交换 三.实验过程 1.完成路由…

链表——循环链表

其他形式的链表——循环链表 循环链表 定义&#xff1a;循环链表是表中最后一个结点的指针指向头结点&#xff0c;使链表构成环状 特点&#xff1a;从表中任一结点发出均可找到表中其他结点&#xff0c;提高查找效率 双向循环链表 data&#xff1a;数据元素 prior&#xff1…

进程间通信--管道

文章目录 一.通信二.管道匿名管道&#xff08;只能用于有血缘关系的进程之间通信&#xff09;1.匿名管道的创建2.匿名管道的读取情况3.管道的特征4.基于匿名管道的简单进程池 有名管道&#xff08;用于没有血缘关系的进程间的通信&#xff09;1.有名管道的建立和删除2.通过一段…

CodeForces 438 D线段树暴力修改

线段树暴力修改 关键是每个数取余操作&#xff0c;看似我们如果暴力改的话m*n肯定超时了 容易发现一个数小于给定模数的时候&#xff0c;取模不会发生改变&#xff0c;而大于给定模数的时候我们得到的最大的结果是x/2向上取整&#xff0c;结果一定小于等于这个数字,这里我想说…

AECC全球留学趋势报告解读

AECC全球留学趋势报告解读 相对更安全的留学目的地&#xff1a; 留学安全是留学生及家长最关注的一个问题。报告显示&#xff0c;在“你认为相对更安全的留学目 的"的问答中&#xff0c;澳大利亚获得总评8.76分&#xff08;满分10分&#xff09;位居第一。 英澳新留学更受…

VMware Tanzu Kubernetes Grid (TKG) 2.1 - 企业级 Kubernetes 解决方案

VMware Tanzu Kubernetes Grid (TKG) 2.1 - 企业级 Kubernetes 解决方案 VMware 构建、签名和支持的开源 Kubernetes 容器编排平台的完整分发版 请访问原文链接&#xff1a;https://sysin.org/blog/vmware-tkg-2/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留出处…

华为OD机试真题 Java 实现【最优资源分配】【2023Q1 200分】

一、题目描述 某块业务芯片最小容量单位为 1.25G&#xff0c;总容量为 M*1.25G&#xff0c;对该芯片资源编号为 1&#xff0c;2&#xff0c;… M。 该芯片支持 3 种不同的配置&#xff0c;分别为 A、B、C. 配置 A: 占用容量为 1.25 * 1 1.25G配置 B: 占用容量为 1.25* 2 2…

白嫖chatgpt的Edge插件,很难不爱啊

目录 &#x1f341;1.常见的Edge浏览器界面 &#x1f341;二.安装WebTab插件 &#x1f341;三.WebTab插件的各种功能 &#x1f341;1.支持免费的chatgpt&#xff0c;不限次数​编辑 &#x1f341;2.有几个休闲的小游戏可以玩耍&#xff0c;点击即玩。 &#x1f341;3.支…

【迷宫问题】找出迷宫所有可能的路径C++

1 引入情境 我记得小时候玩过推箱子游戏&#xff0c;也是如下图这种&#xff0c;四周由深色的方格作为墙壁&#xff0c;白色的地方是可以通过的。现在想要从红色方格出发走到黄色方格&#xff0c;能有什么好办法呢&#xff1f; 注意到&#xff0c;对于计算机没有全局的观念&…

Flutter音乐播放audioplayers

简介 Flutter的audioplayers是一个Flutter插件&#xff0c;可以播放多个同时的音频文件&#xff0c;支持Android、iOS、Linux、macOS、Windows和web平台。它有以下特点&#xff1a; 可以从本地文件、网络资源或内存中加载音频可以控制音量、进度、速度和循环可以播放多个音频…

《编程思维与实践》1069.第一位数字

《编程思维与实践》1069.第一位数字 题目 思路 由于正整数N的N次方最大可以为 1 0 8 ⋅ 1 0 8 10^{8\cdot 10^8} 108⋅108,加上数据可能有很多组,所以直接采用大整数计算次方这方法很可能超时, 这里给出一种数学算法: 幂指函数通常的处理方式是取对数将乘方转化为乘法: N N 1…

Linux驱动编程(驱动程序基石)(上)

一、休眠与唤醒 要休眠的线程&#xff0c;放在 wq 队列里&#xff0c;中断处理函数从 wq 队列里把它取出来唤醒。所以&#xff0c;我们要做这几件事&#xff1a; ① 初始化 wq 队列 ② 在驱动的 read 函数中&#xff0c;调用 wait_event_interruptible&#xff1a; 它本身会判断…

English Learning - L3 作业打卡 Lesson1 Day7 2023.5.11 周四

English Learning - L3 作业打卡 Lesson1 Day7 2023.5.11 周四 引言&#x1f349;句1: Blues is slow, sad and soulful.成分划分弱读连读爆破语调 &#x1f349;句2: Duke Ellington and his orchestra recorded a famous song – Mood Indigo – about the deep blue color, …

【嵌入式环境下linux内核及驱动学习笔记-(11-设备树)】

目录 1、设备树体系1.1 DTS /DTSI / DTC / DTB 2、基础语法2.1 节点语法2.1.1 通用名称建议 2.2 属性语法2.2.1 属性值 2.3 关于label2.4 节点的[unit-address] 与reg属性2.5 根节点 /2.6 标准属性compatible2.6.1 of_machine_is_compatible函数 2.7 地址编码2.7.1 标准属性#ad…

Linux常用命令(4)

文章目录 Linux常用命令(4)查找文件或目录 find在当前文件下找一个hello.txt的文件在/user/include目录下找stdio.h文件 查找内容 grep在当前目录的hello.txt 文件中搜索1234在当前目录下所有文件中搜索1234在当前目录下所有以.txt结尾的文件中搜索1234 制作压缩包和解压缩命令…

Java 函数式编程 详细介绍

在兼顾面向对象特性的基础上&#xff0c;Java语言通过Lambda表达式与方法引用等&#xff0c;为开发者打开了函数式编程的大门。 下面我们做一个初探。 Lambda的延迟执行 有些场景的代码执行后&#xff0c;结果不一定会被使用&#xff0c;从而造成性能浪费。而Lambda表达式是延…

PID现场参数调试解密

看图 反馈与给定曲线 1.超调过大&#xff0c;减小比例&#xff0c;增大积分时间 2.迅速变化&#xff0c;存在小超调 3.实际值缓慢接近设定值&#xff0c;并且无超调的达到设定值 4.增益系数太小和/或微分时间太长 5.益系数太小和/或积分时间太长 公式(西门子FB41) 举例 现象…

20230513查找瑞芯微RK3588开发板以及对DP接口的支持

20230513查找瑞芯微RK3588开发板以及对DP接口的支持 2023/5/13 17:43 01、t-firefly https://www.t-firefly.com/ https://www.t-firefly.com/product/industry/aio1684xjd4 https://www.t-firefly.com/product/industry/aio3588q https://item.taobao.com/item.htm?spma1z10.…

三轴和直剪

目录 常规三轴 成样 预压 围压加载 二维直剪 成样 预压 围压加载 常规三轴 成样 cylinder keyword ...(3D ONLY) Generate a cylinder in 3D. If the name keyword has not been specified, then s cylinderWall. By default, the cylinder is closed and each side o…