数据结构--栈与队列

目录


前言

1.栈

1.1栈的概念及结构

 1.2接口函数

 1.3函数实现

1.4如何使用

2.队列 

2.1队列的概念及结构

2.2接口函数

 2.3函数实现

2.4如何使用


前言

        前面我们已经学习了顺序表和链表,今天我们来学习栈与队列,这两种结构也属于线性表,实际上就是顺序表和链表结构的延展。


1.栈


1.1栈的概念及结构

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


 1.2接口函数

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

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

// 下面是定长的静态栈的结构,实际中一般不实用,所以我们主要实现下面的支持动态增长的栈
typedef int STDataType;

// 支持动态增长的栈
typedef int STDataType;
typedef struct Stack
{
	STDataType* _a;
	int _top; // 栈顶
	int _capacity; // 容量
}Stack;
// 初始化栈
void StackInit(Stack* ps);
// 入栈
void StackPush(Stack* ps, STDataType data);
// 出栈
void StackPop(Stack* ps);
// 获取栈顶元素
STDataType StackTop(Stack* ps);
// 获取栈中有效元素个数
int StackSize(Stack* ps);
// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0
bool StackEmpty(Stack* ps);
// 销毁栈
void StackDestroy(Stack* ps);

 1.3函数实现

1.单个节点成员介绍

typedef struct Stack
{
	STDataType* _a;
	int _top; // 栈顶
	int _capacity; // 容量
}Stack;

        在这里我们使用顺序表来实现栈结构,当然使用单链表,双向链表也可以。我们可以看到有成员top,它是用来指向栈顶元素的,当然也可以指向栈顶元素的下一个,这取决于你的实现逻辑。这里的顺序表,显然也是动态的。


2.初始化栈

void StackInit(Stack* ps)
{
	assert(ps);
	ps->_a = NULL;
	ps->_capacity = NULL;
	ps->_top = 0;

}

        关于top的初始化值的问题。

 这里可以将top初识化为0,当然也可以将top初始化为-1。这两种逻辑有何不同呢?

        1.将top初识化为0,假如在top位置插入一个数据,那么top就要向后移动一位。如果top不像后移动一位,那么就区分不了0位置是有数据还是没有数据了。所以top==0时,top表示的是指向栈顶元素的后一位。        

        2.将top初识化为-1,假如在top位置插入一个数据当然也是要向后面移动一位 。所以当top==-1时,top表示的时指向栈顶元素。

        在这里我是将top初始化为0了。


3.入栈

void StackPush(Stack* ps, STDataType data)
{
	assert(ps);
	if (ps->_top == ps->_capacity)
	{
		int newcapacity = ps->_capacity == 0 ? 4 : 2 * ps->_capacity;
		Stack* tmp = (Stack*)realloc(ps->_a, sizeof(Stack) * newcapacity);
		if (tmp == NULL)
		{
			perror("realloc");
			return;
		}
		ps->_a = tmp;
		ps->_capacity = newcapacity;
	}
	
	ps->_a[ps->_top] = data;
	ps->_top++;
}

        要实现入栈操作是十分简单的,首先要创建一个新的节点,在这扩容操作没必要单独封装,因为只有入栈操作时才用的到。不要忘记断言。

        插入操作是十分简单的,只需要在top位置插入,再让top指针向后移动就好了。


4.出栈

void StackPop(Stack* ps)
{
	assert(ps);
	assert(ps->_top > 0);
	ps->_top--;
}

        出栈操作也是十分简单,只需要让top向前移动一位就好了。


5.返回栈顶元素

STDataType StackTop(Stack* ps)
{
	assert(ps);
	assert(ps->_top > 0);
	return ps->_a[ps->_top - 1];
}

        进行栈顶元素返回操作的时候,需要注意的是,要将top-1,因为top是指向栈顶元素的下一个位置。


6.获取栈中有效元素个数

int StackSize(Stack* ps)
{
	assert(ps);
	return ps->_top;
}

        直接返回top的值就好了,在此逻辑种top值就是元素的个数。


7.检测栈是否为空

bool StackEmpty(Stack* ps)
{
	assert(ps);
	return ps->_top == 0;//等于0为真,否则假
}

        在此逻辑下,top值为空就代表栈为空,所以只需要判断top是否为0就好了,真则返回true,假则返回false。


8.销毁栈

void StackDestroy(Stack* ps)
{
	assert(ps);
	free(ps->_a);
	ps->_capacity = 0;
	ps->_top = 0;
}

        因为是用顺序表实现栈的,所以直接free掉malloc出来的空间就好了。

完!!!


1.4如何使用

 

int main()
{
	Stack s;
	StackInit(&s);
	StackPush(&s, 1);
	StackPush(&s, 2);

	StackPush(&s, 3);
	
	printf("栈内又%d个数据\n", StackSize(&s));
	while (!StackEmpty(&s))
	{
		printf("%d\n", StackTop(&s));
		StackPop(&s);
	}
	printf("\n");
	StackDestroy(&s);
	return 0;
}

        我们要获取栈内元素时,可以利用循环操作,直到栈内数据为空,要注意的是,每获取一次栈顶元素时,要进行出栈操作,才能取到下一个数据。

        值得一提的是:如果入栈时,在数据没有全部入栈时,突然出栈一个数据,然后在进行入栈,最后数据的顺序会被打乱

int main()
{
	Stack s;
	StackInit(&s);
	StackPush(&s, 1);
	StackPush(&s, 2);
	printf("%d\n", StackTop(&s));
	StackPop(&s);
	StackPush(&s, 3);
	
	printf("栈内又%d个数据\n", StackSize(&s));
	while (!StackEmpty(&s))
	{
		printf("%d\n", StackTop(&s));
		StackPop(&s);
	}
	printf("\n");
	StackDestroy(&s);
	return 0;
}


2.队列 


2.1队列的概念及结构

队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出FIFO(First In First Out)

入队列:进行插入操作的一端称为队尾

出队列:进行删除操作的一端称为队头


2.2接口函数

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

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
// 链式结构:表示队列
typedef int QDataType;
typedef struct QListNode
{
	struct QListNode* _pNext;
	QDataType _data;
}QNode;
// 队列的结构
typedef struct Queue
{
	QNode* _front;
	QNode* _rear;
	int size;
}Queue;
// 初始化队列
void QueueInit(Queue* q);
// 队尾入队列
void QueuePush(Queue* q, QDataType data);
// 队头出队列
void QueuePop(Queue* q);
// 获取队列头部元素
QDataType QueueFront(Queue* q);
// 获取队列队尾元素
QDataType QueueBack(Queue* q);
// 获取队列中有效元素个数
int QueueSize(Queue* q);
// 检测队列是否为空,如果为空返回非零结果,如果非空返回0
bool QueueEmpty(Queue* q);
// 销毁队列
void QueueDestroy(Queue* q);

 2.3函数实现

1.单个节点成员介绍

typedef struct QListNode
{
	struct QListNode* _pNext;
	QDataType _data;
}QNode;
// 队列的结构
typedef struct Queue
{
	QNode* _front;
	QNode* _rear;
	int size;
}Queue;

        

        实现队列我们使用头尾两个指针是最优的,但为什么,不在实现函数内直接定义,而直接在结构体内封装头和尾呢?这样做可以避免二级指针发的使用。如果队列为空,第一次进行入队操作时,就涉及到要改变结构体指针的操作,这时我们就要用到结构体指针的指针,也就是二级指针。但如果,像这样封装一下,就起到了哨兵位的作用就不用传二级指针了。


2.初始化队列

void QueueInit(Queue* q)
{
	assert(q);
	q->_front=q->_rear = NULL;
	
	q->size = 0;
}

3.队尾入队列

void QueuePush(Queue* q, QDataType data)
{
	assert(q);
	QNode* newnode = (QNode*)malloc(sizeof(QNode));
	if (newnode == NULL)
	{
		perror("malloc fail");
		return;
	}
	newnode->_data = data;
	newnode->_pNext = NULL;
	if (q->_rear == NULL)
	{
		q->_front = q->_rear = newnode;

	}
	else
	{
		q->_rear->_pNext = newnode;
		q->_rear = newnode;
	}
	q->size++;
}

        入队列操是十分简单的,只需要利用队尾指针添加新节点就好了。


4.队头出队列

void QueuePop(Queue* q)
{
	assert(q);
	assert(q->_front);
	QNode* cur = q->_front;
	q->_front = q->_front->_pNext;
	free(cur);
	cur = NULL;
	if (q->_front == NULL)
		q->_rear = NULL;
		
	q->size--;
}

        在进行出队操作时,注意保存队头的下一个节点,然后free出队。要注意的是,队列数据为空,就不能进行出队操作了,可以进行断言assert(q->_front)防止。当只剩一个节点的时候,头尾指针指向一个节点,这是就要判断队头是否为空,如果为空,那也要将尾指针置空,防止野指针。


5.获取队列头部元素

QDataType QueueFront(Queue* q)
{
	assert(q);
	assert(q->_front);
	return q->_front->_data;
	
}

6.获取队列尾部元素

QDataType QueueBack(Queue* q)
{
	assert(q);
	assert(q->_rear);
	return q->_rear->_data;
}

7.获取队列中有效元素个数

int QueueSize(Queue* q)
{
	assert(q);
	return q->size;
}

8.检查队列是否为空

bool QueueEmpty(Queue* q)
{
	assert(q);
	return q->_front == NULL;
}

        这里只需要检查对头是否为空就好了。


9.销毁队列

void QueueDestroy(Queue* q)
{
	assert(q);
	QNode* cur = q->_front;
	while (cur)
	{
		QNode* next = cur->_pNext;
		free(cur);
		cur = next;
	}
	q->_front = q->_rear = NULL;
	q->size = 0;
}

        这里我们依然是保存后一个,销毁当前节点,所有节点都销毁后,将头尾指针置空。

完!!!


2.4如何使用

int main()
{
	Queue s;
	QueueInit(&s);
	QueuePush(&s, 1);
	QueuePush(&s, 2);
	QueuePush(&s, 3);
	QueuePush(&s, 4);
	QueuePush(&s, 5);
	printf("%d个元素\n", QueueSize(&s));
	
	while (!QueueEmpty(&s))
	{
		printf("%d ", QueueFront(&s));
		QueuePop(&s);
	}
	QueueDestroy(&s);
	
	return 0;
}

        这里我们也是利用循环,将所有数据获取,获取一个数据,出一个数据。

值得一提的是:如果入队时,在数据没有全部入栈时,突然出队一个数据,然后在进行入栈,最后数据的顺序也不会被打乱

int main()
{
	Queue s;
	QueueInit(&s);
	QueuePush(&s, 1);
	QueuePush(&s, 2);
	QueuePush(&s, 3);
	QueuePush(&s, 4);
	printf("%d\n", QueueFront(&s));
	QueuePop(&s);
	QueuePush(&s, 5);
	printf("%d个元素\n", QueueSize(&s));
	
	while (!QueueEmpty(&s))
	{
		printf("%d ", QueueFront(&s));
		QueuePop(&s);
	}
	QueueDestroy(&s);
	
	return 0;
}

希望这篇文章对你有帮助!!!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

解决windeployqt打包exe的“VCINSTALLDIR is not set“问题

今天在使用windeployqt部署qt的.exe文件时&#xff0c; 出现如下错误&#xff1a; windeployqt HelloQt.exe图(1) 报"VCINSTALLDIR路径"找不到 出现这种情况的原因是&#xff1a;VCINSTALLDIR环境没有配置&#xff0c;需要把Visual Studio的编译路径: ## 1) 社区版…

2018年五一杯数学建模A题徐州潘安湖风景区游览路线设计解题全过程文档及程序

2019年五一杯数学建模 A题 徐州潘安湖风景区游览路线设计 原题再现 徐州是一个老工业基地和资源型城市&#xff0c;煤炭开采历史长达130年。长期煤炭开采在徐州累计形成采煤塌陷区达数十万亩。位于徐州市贾汪区西南部、紧邻马庄的潘安湖湿地公园原来就是徐州最大的、塌陷最严…

关系代数、SQL语句和Go语言示例

近些年&#xff0c;数据库领域发展日新月异&#xff0c;除传统的关系型数据库外&#xff0c;还出现了许多新型的数据库&#xff0c;比如&#xff1a;以HBase、Cassandra、MongoDB为代表的NoSQL数据库&#xff0c;以InfluxDB、TDEngine为代表的时序数据[1]库&#xff0c;以Neo4J…

【UE5】物体沿样条线移动

目录 效果 步骤 一、使用样条线创建路径 二、创建沿样条线路径移动的物体 三、定义可移动物体的生成器 效果 步骤 一、使用样条线创建路径 先创建一个Actor蓝图&#xff0c;这里命名为“BP_Line” 该蓝图中只需添加一个样条组件 将“BP_Line”拖入场景中 按住Alt鼠标左键…

生存分析后如何绘制亚组森林图?小白也能快速搞定!(附教程)

本周为大家重点介绍一下风暴统计平台的最新板块——亚组森林图&#xff01; 现在亚组分析好像越来越流行&#xff0c;无论是观察性研究还是RCT研究&#xff0c;亚组分析一般配备森林图。 比如这张图&#xff1a; 还有这个&#xff1a; 森林图不仅是画图的画法&#xff0c;背后还…

Javaweb之Vue指令的详细解析

2.3 Vue指令 在上述的快速入门中&#xff0c;我们发现了html中输入了一个没有学过的属性v-model&#xff0c;这个就是vue的指令。 指令&#xff1a;HTML 标签上带有 v- 前缀的特殊属性&#xff0c;不同指令具有不同含义。例如&#xff1a;v-if&#xff0c;v-for… 在vue中&a…

Zookeeper Java 开发,自定义分布式锁示例

文章目录 一、概述二、导入依赖包三、创建锁的过程3.1 通过 create 创建节点信息3.2 AsyncCallback.StringCallback 回调函数3.3 AsyncCallback.Children2Callback 的回调函数3.4 Watcher 的回调函数 四、完整示例4.1 完整分布式锁代码4.2 测试类 如果您还没有安装Zookeeper请看…

第四章 串【24王道数据结构笔记】

1.串的基本概念 串&#xff0c;即字符串 (String) 是由零个或多个字符组成的有限序列。一般记为Sa1a2.....an(n>0) S"HelloWorld!" TiPhone 11 Pro Max? 其中&#xff0c;S是串名&#xff0c;单引号括起来的字符序列是串的值;a;可以是字母、数字或其他字符;串中…

智能售货柜:小本投资的不二之选

智能售货柜&#xff1a;小本投资的不二之选 智能售货柜的运营优势在于&#xff1a;一是降低运营成本&#xff0c;不需要大量员工&#xff1b;二是具备自动识别和智能结算功能&#xff0c;提高运营效率&#xff1b;三是提供数据分析&#xff0c;优化产品和服务。相比传统零售店&…

初学UE5 C++②

目录 导入csv表格数据 创建、实例化、结构体 GameInstance Actor camera 绑定滚轮控制摇臂移动 碰撞绑定 角色碰撞设定 按钮 UI显示 单播代理 多播和动态多播 写一个接口 其他 NewObject 和 CreateDefaultSubobject区别 导入csv表格数据 创建一个object的C类 …

怎样备份电脑文件比较安全

域智盾软件是一款功能强大的电脑监控软件&#xff0c;它不仅具备实时屏幕监控、行为审计等功能&#xff0c;还能够对电脑文件进行备份和管理。下面将介绍域智盾软件如何备份电脑文件&#xff0c;以确保数据安全。 1、开启文档备份功能 部署后台&#xff0c;然后点击文档安全&a…

30天黑客(网络安全)自学

前言 前几天发布了一篇 网络安全&#xff08;黑客&#xff09;自学 没想到收到了许多人的私信想要学习网安黑客技术&#xff01;却不知道从哪里开始学起&#xff01;怎么学 今天给大家分享一下&#xff0c;很多人上来就说想学习黑客&#xff0c;但是连方向都没搞清楚就开始学习…

科技创新 共铸典范 | 江西卫健办邓敏、飞图影像董事长洪诗诗一行到访拓世科技集团,提振公共卫生事业发展

2023年11月15日&#xff0c;拓世科技集团总部迎来了江西省卫健项目办项目负责人邓敏、江西飞图影像科技有限公司董事长洪诗诗一行的考察参观&#xff0c;集团董事长李火亮、集团高级副总裁方高强进行热情接待。此次多方交流&#xff0c;旨在共同探讨携手合作&#xff0c;激发科…

Win7安装nvme协议的SSD硬盘方法

自家用的电脑硬盘不够用&#xff0c;于是想买块硬盘扩展下存储。市面上&#xff0c;我比较了下SSD&#xff0c;一类是原来的SATA协议的固态硬盘&#xff0c;一类是M2的固态硬盘&#xff0c;我发现SATA的硬盘比M2的贵&#xff0c;我的主板较老&#xff0c;又不没有原生支持M2的接…

Python---列表 集合 字典 推导式(本文以 列表 为主)

推导式&#xff1a; 推导式comprehensions&#xff08;又称解析式&#xff09;&#xff0c;是Python的一种独有特性。推导式是可以从一个数据序列构建另一个新的数据序列&#xff08;一个有规律的列表或控制一个有规律列表&#xff09;的结构体。 共有三种推导&#xff1a;列表…

windows监控打印机状态工具

windows监控打印机状态工具 实时监控打印机状态&#xff0c;打印总页数&#xff0c;以及打印故障提醒。 工具下载地址

《硅基物语.AI写作高手:从零开始用ChatGPT学会写作》《从零开始读懂相对论》

文章目录 《硅基物语.AI写作高手&#xff1a;从零开始用ChatGPT学会写作》内容简介核心精华使用ChatGPT可以高效搞定写作的好处如下 《从零开始读懂相对论》内容简介关键点书摘最后 《硅基物语.AI写作高手&#xff1a;从零开始用ChatGPT学会写作》 内容简介 本书从写作与ChatG…

ORB SLAM3 使用二进制文件 ORBvoc.bin 加载Vocabulary

使用 二进制文件 ORBvoc.bin 加载Vocabulary&#xff0c;将比ORBvoc.txt 速度快很多倍&#xff01; 实测1秒内完成加载&#xff1a; 一、下载ORBvoc.bin 百度网盘&#xff1a; ORBvoc.bin下载链接 提取码&#xff1a;dyyk 解压后&#xff0c;将ORBvoc.bin拷贝到Vocabulary文…

5G与中国的海

今年国庆假期&#xff0c;香港迎来了阔别5年的国庆维港烟花汇演 10月1日晚上9点&#xff0c;“HKT x FWD 2023 年国庆烟花汇演”在维多利亚港上空上演。在23分钟时间里&#xff0c;燃放了超过3万枚烟花。而与以往维港烟花秀不同的是&#xff0c;为了让更多民众欣赏这次表演&…

【C++面向对象】15. 模板

文章目录 【 1. 函数模板 】【 2. 类模板 】 模板是泛型编程的基础&#xff0c;泛型编程即以一种独立于任何特定类型的方式编写代码。模板是指创建泛型类或函数的蓝图或公式。库容器&#xff0c;比如迭代器和算法&#xff0c;都是泛型编程的例子&#xff0c;它们都使用了模板的…