数据结构:栈和队列(Stack篇)(简单易懂超详细)
创作时间:
作者:
@小白创作中心
数据结构:栈和队列(Stack篇)(简单易懂超详细)
引用
CSDN
1.
https://blog.csdn.net/Jdxxwu/article/details/142314129
栈是数据结构中一种重要的线性表,其只允许在固定的端进行插入和删除元素操作。本文将详细介绍栈的概念、实现方式以及相关操作的代码示例。
前言
今天我们一起来实现数据结构中的栈。
一、栈是什么?
1. 栈的概念
栈:一种特殊的线性表,其只允许在固定的端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。
压栈:栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶。
出栈:栈的删除操作叫做出栈。出数据也在栈顶。
栈就像是一叠盘子,我们放盘子只能放到盘子的最上面,同理取盘子,也只能从栈的最上方来取。放盘子称为入栈,去盘子称为出栈。
2.栈的结构选择
栈底层结构选型:
栈的实现一般可以使用数组或者链表实现,相对而言数组的结构实现更优一些。因为数组在尾上插入数据的代价比较小。
下面我们来对比一下这几种实现:
二、栈的实现
1. 栈结构体的定义
基于下面这张图,再类比顺序表,我们可以发现,栈的实现需要三个元素。
typedef int STDataType;
typedef struct Stack
{
STDataType* arr; //栈数组
int capacity; //栈的空间大小
int top; //栈顶位置
}ST;
2. 栈的初始化
//栈的初始化
void STInit(ST* ps);
我们需要将arr , capacity , top这三个元素初始化。具体的实现是:
//栈的初始化
void STInit(ST* ps)
{
assert(ps);
ps->arr = NULL;
ps->capacity = 0;
ps->top = 0;
}
3. 栈的销毁
有初始化必然有销毁。
//栈的销毁
void STDestroy(ST* ps);
我们需要释放栈中的空间,别忘了将其置为NULL。释放掉栈中所有的元素。
//栈的销毁
void STDestroy(ST* ps)
{
assert(ps);
if(ps->arr)
free(ps->arr);
ps->arr = NULL;
ps->capacity = 0;
ps->top = 0;
}
3. 入栈
//数据入栈
void StackPush(ST* ps, STDataType x); //STDataType x 是我们需要插入的数据
在入栈之前我们需要判断数组容量够不够,需不需要扩容。具体实现是
//数据入栈
void StackPush(ST* ps, STDataType x)
{
assert(ps);
if (ps->capacity == ps->top) //空间满了需要扩容
{
int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity; //三目运算符如果原本栈为空,就赋初始为4个空间,若不为空,则双倍扩容
STDataType* tem = (STDataType*)realloc(ps->arr, newcapacity * sizeof(STDataType));
//判断所开空间是否成功
if (tem == NULL)
{
perror("realloc fail!");
exit(1);
}
ps->arr = tem;
ps->capacity = newcapacity;
}
//入栈开始
ps->arr[ps->top++] = x;
}
4.出栈
注意,如果栈为空,则不能出数据。因此先写判空方法。注意包含头文件
#pragma once #include<stdio.h> #include<stdlib.h> #include<assert.h> #include<stdbool.h>
//数据出栈
void StackPop(ST* ps);
//栈判空
bool StackEmpty(ST* ps)
{
assert(ps);
return ps->top == 0;
}
开始出栈
//数据出栈
void StackPop(ST* ps)
{
assert(ps);
assert(!StackEmpty(ps));
ps->top--;
}
5. 取栈顶元素
//取栈顶元素
STDataType StackTop(ST* ps);
//取栈顶元素
STDataType StackTop(ST* ps)
{
assert(ps);
assert(!StackEmpty(ps));
return ps->arr[ps->top - 1];
}
那么只要栈不为空,就可以一直取栈顶元素
6. 栈中元素的个数
//获取栈中有效元素个数
int STSize(ST* ps);
//获取栈中有效元素个数
int STSize(ST* ps)
{
assert(ps);
return ps->top;
}
总结
栈我们就实现完了,下面是栈实现的完整代码,需要的小伙伴们自取~
//Stack.h
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
typedef int STDataType;
typedef struct Stack
{
STDataType* arr; //栈数组
int capacity; //栈的空间大小
int top; //栈顶位置
}ST;
//栈的初始化
void STInit(ST* ps);
//栈的销毁
void StackDestroy(ST* ps);
//数据入栈
void StackPush(ST* ps, STDataType x); //STDataType x 是我们需要插入的数据
//数据出栈
void StackPop(ST* ps);
//取栈顶元素
STDataType StackTop(ST* ps);
//获取栈中有效元素个数
int STSize(ST* ps);
//Stack.c
#include"Stack.h"
//栈的初始化
void STInit(ST* ps)
{
assert(ps);
ps->arr = NULL;
ps->capacity = 0;
ps->top = 0;
}
//栈的销毁
void STDestroy(ST* ps)
{
assert(ps);
if(ps->arr)
free(ps->arr);
ps->arr = NULL;
ps->capacity = 0;
ps->top = 0;
}
//数据入栈
void StackPush(ST* ps, STDataType x)
{
assert(ps);
if (ps->capacity == ps->top) //空间满了需要扩容
{
int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity; //三目运算符如果原本栈为空,就赋初始为4个空间,若不为空,则双倍扩容
STDataType* tem = (STDataType*)realloc(ps->arr, newcapacity * sizeof(ST));
//判断所开空间是否成功
if (tem == NULL)
{
perror("realloc fail!");
exit(1);
}
ps->arr = tem;
ps->capacity = newcapacity;
}
//入栈开始
ps->arr[ps->top++] = x;
}
//栈判空
bool StackEmpty(ST* ps)
{
assert(ps);
return ps->top == 0;
}
//数据出栈
void StackPop(ST* ps)
{
assert(ps);
assert(!StackEmpty(ps));
ps->top--;
}
//取栈顶元素
STDataType StackTop(ST* ps)
{
assert(ps);
assert(!StackEmpty(ps));
return ps->arr[ps->top - 1];
}
//获取栈中有效元素个数
int STSize(ST* ps)
{
assert(ps);
return ps->top;
}
//测试样例
#include"Stack.h"
void STTest()
{
ST st;
STInit(&st);
//
StackPush(&st, 1);
StackPush(&st, 2);
StackPush(&st, 3);
StackPush(&st, 4);
printf("size: %d\n", STSize(&st));//4
//StackPush(&st, 5);
//StackPop(&st);
//循环出栈,直到栈为空
while (!StackEmpty(&st))
{
STDataType data = StackTop(&st);
printf("%d ", data);
//出栈
StackPop(&st);
}
printf("size: %d\n", STSize(&st));//0
///
STDestroy(&st);
}
int main()
{
STTest();
return 0;
}
热门推荐
尿酸高不只害痛风「风吹就会痛」!简鈺樺营养师:高尿酸隐藏4大类疾病风险
月季花的功效与作用(食用/护肤/观赏/药用)
如何让癌症营养补充品更有效?专家建议:需搭配营养师
2025 年商标续展全攻略:流程、费用与常见问题解答
和光同尘,与时舒卷
求职防骗指南 别让这些套路“坑”了你
项目质量和制程质量区别
天然气压裂返排液与石油压裂返排液的区别
肝内偏强回声怎么回事
生物惰性液相质谱联用系统直接进样法分析食品中草甘膦和氨甲基膦酸
拉萨十大特色美食,你都吃过几种!
研究发现:胰腺癌初期不是口渴多尿,而是悄悄出现这3个小信号
公职人员是什么?从定义到职责的全面解析
如何练就徒手写出千行Verilog代码的能力
服务供给开新篇丨把家政做成“家门口的民生工程”
黄金的购买途径和注意事项有哪些?如何选择合适的购买渠道?
摩托车撞人了怎么处理
2025年中国收音机行业发展历程、产业链图谱、发展现状及未来前景分析
春季,警惕抑郁症高发!这份心理健康指南请收好
西湖大学科学家在大脑中“定位”抑郁症
為什麼結婚戒指要戴在無名指上?其實背後有這些文化意義!
“橘生淮南则为橘”:从典故到现代应用的文化解读
电脑开机黑屏怎么办?六步排查法帮你轻松应对
解决音频文件无法播放问题的实用指南分享
过生日是过农历的,还是阴历的?生日不能随便过,这些讲究要知道
孩子过生日,到底选“阳历还是阴历”?知道这些讲究,不纠结
汽车倒车时为什么会发出咯噔声
倒车雷达声音与距离,倒车雷达怎么判断距离
曼联2028年英超夺冠展望:雄心与现实的多维博弈
玉米期货价格波动的影响因素及应对策略