【数据结构】带你图文结合深入栈和队列,并具体分步实现

这篇具有很好参考价值的文章主要介绍了【数据结构】带你图文结合深入栈和队列,并具体分步实现。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【数据结构】带你图文结合深入栈和队列,并具体分步实现,初阶数据结构,数据结构,算法,c语言,c++,开发语言

君兮_的个人主页

勤时当勉励 岁月不待人

C/C++ 游戏开发

Hello,米娜桑们,这里是君兮_,我们继续来学习初阶数据结构的内容,今天我们要讲的是栈与队列部分的内容,这篇博客先讲栈,队列我们放到下次再讲
好了,废话不多说,开始今天的学习吧!

一.栈

1.栈的概念及结构

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

  • 压栈:栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶。
    出栈:栈的删除操作叫做出栈。出数据也在栈顶
    【数据结构】带你图文结合深入栈和队列,并具体分步实现,初阶数据结构,数据结构,算法,c语言,c++,开发语言

  • 无论是入栈还是出栈,都遵循后进先出原则。

2.栈的实现方式

  • 栈的实现一般可以使用数组或者链表实现,相对而言数组的结构实现更优一些。因为数组在尾上插入数据的不需要付出很大的代价。
    【数据结构】带你图文结合深入栈和队列,并具体分步实现,初阶数据结构,数据结构,算法,c语言,c++,开发语言
  • 如果使用链表方式实现的话,会出现一个问题,我们的栈实际上是通过尾插实现入栈的,也就是说在入栈时我们每一次都需要通过遍历的方式找尾,或者使用双头循环链表的方式找尾,但是由于数组的连续性和支持随机访问,对比于链表无疑是更加方便的。因此,接下来内容中的栈结构我们就通过数组的方式实现。

3.栈的分类

  • 栈分为静态栈和动态栈

静态栈

// 下面是定长的静态栈的结构,实际中一般不实用
typedef int STDataType;
#define N 10
typedef struct Stack
{
STDataType a[N];
int _top; // 栈顶
int _capacity; // 容量
}Stack;

动态栈

  • 动态栈是通过动态内存开辟的方式实现的,由于支持动态内存的变化,因此在实际应用中动态栈无疑是更好的选择
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);
//获取栈底元素
STDataType StackTail(Stack* ps);

4.动态栈的实现

栈的初始化函数 StackInit

  • 无论是什么数据结构,我们都得先初始化
// 初始化栈
void StackInit(Stack* ps)
{
    assert(ps);//断言
    ps->a = NULL;//数组中还没有元素
    //容量和有效元素都为0
    ps->capacity = 0;
    ps->top = 0;
}

入栈函数 StackPush

// 入栈
void StackPush(Stack* ps, STDataType data)
{
    assert(ps);
    //判断是否还有空间
    if (ps->capacity == ps->top)
    {
        int newCapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;//通过容量判断此时栈中有没有元素,如果没有元素就申请4个字节的空间,如果有容量,但容量不够时就把容量变为之前的2倍
        STDataType* tmp = (STDataType*)realloc(ps->a, newCapacity* sizeof(STDataType));
        if (tmp == NULL)//判读开辟空间是否成功
        {
            perror("realloc failed");
            exit(-1);
        }
        ps->a = tmp;//成功了就把这个新的空间给我们的a数组
        ps->capacity = newCapacity;//容量也要做相应的动态变化
    }
    ps->a[ps->top] = data;//往栈中存元素
    ps->top++;//有效元素++
}

【数据结构】带你图文结合深入栈和队列,并具体分步实现,初阶数据结构,数据结构,算法,c语言,c++,开发语言

  • 这里可能有些人对于动态内存管理的内容不够了解,可以看看我这篇博客:【C语言进阶】那些你必须掌握的C/C++要点——动态内存管理(1)
  • 这里还有一点要提,这里为什么用realloc开辟空间而不是使用malloc呢?
  • 这里就涉及一些realloc的进阶玩法
  • 我们知道realloc是用来修改动态开辟的内存大小,但是当我们给它传入的是一个空指针,它的功能与malloc就是相同的。

出栈的函数 StackPop

// 出栈
void StackPop(Stack* ps)
{
    assert(ps);
    --ps->top;
}
  • 出栈就非常简单啦,我们直接让top减1就行,由于我们的数组是通过下标的方式访问数组成员的,top只有减少就无法在找到相应的元素啦
    【数据结构】带你图文结合深入栈和队列,并具体分步实现,初阶数据结构,数据结构,算法,c语言,c++,开发语言

获取栈顶元素的函数 StackTop

// 获取栈顶元素
STDataType StackTop(Stack* ps)
{
    assert(ps);
    assert(ps->top > 0);
    //有效元素为top,数组的下标得-1
    return ps->a[ps->top-1];
}
  • 我们栈中的元素是尾插的,因此最后一个元素就是我们的栈顶元素

获取栈底元素 StackTail

  • 需要栈底的情况其实并不常见,我们会在某些特别情况下才会用到(比如我们之后会带大家写的oj题)
STDataType StackTail(Stack* ps)
{
   assert(ps);
   return ps->a[0];
}
  • 我们的元素是尾插进数组的,因此我们的首元素就是我们的栈底元素。

获取栈中有效元素的个数的函数 StackSize

// 获取栈中有效元素个数
int StackSize(Stack* ps)
{
    assert(ps);
    return ps->top;
}
  • 我们在之前无论是入栈还是出栈都变化了我们的有效元素top,因此返回的top就是我们的有效元素个数。

判断栈是否为空的函数 StackEmpty

// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
bool StackEmpty(Stack* ps)
{
    assert(ps);
    return ps->top == 0;
}
  • top存着我们的有效元素,因此可以通过判断top是否为0来判断栈是否为空

销毁栈的函数

// 销毁栈
void StackDestroy(Stack* ps)
{
    assert(ps);
    free(ps->a);//free掉动态开辟的数组
    ps->a = NULL;//置空
    ps->top = ps->capacity = 0;//把有效元素和容量全部清空
}
  • 我们的内存是通过动态开辟出来的,因此当我们使用完后就必须把我们的申请的内存给free掉防止内存泄漏。

5.测试栈

  • 我们来试试的效果
void TestStack1()
{
    Stack ST;
    StackInit(&ST);
    StackPush(&ST, 10);
    StackPush(&ST, 20);
    StackPush(&ST, 30);
    while (!StackEmpty(&ST))
    {
        printf("%d ", StackTop(&ST));
        StackPop(&ST);
}
    
    StackDestroy(&ST);
   

}
int main()
{
    
    TestStack1();
    return 0;
}

【数据结构】带你图文结合深入栈和队列,并具体分步实现,初阶数据结构,数据结构,算法,c语言,c++,开发语言

总结

  • 今天的内容到这里就结束了,如果你能理解之前讲过的顺序表,链表等,栈的内容其实非常的简单,想学好的话,一定要自己动手试试哦!!
  • 好了,如果你有任何疑问欢迎在评论区或者私信我提出,大家下次再见啦!

新人博主创作不易,如果感觉文章内容对你有所帮助的话不妨三连一下这个新人博主再走呗。你们的支持就是我更新的动力!!!

**(可莉请求你们三连支持一下博主!!!点击下方评论点赞收藏帮帮可莉吧)**

【数据结构】带你图文结合深入栈和队列,并具体分步实现,初阶数据结构,数据结构,算法,c语言,c++,开发语言文章来源地址https://www.toymoban.com/news/detail-638246.html

到了这里,关于【数据结构】带你图文结合深入栈和队列,并具体分步实现的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【数据结构】带你深入理解栈

    栈是一种特殊的线性表。其只允许在固定的一端进行插入和删除元素的操作,进行数据的插入和删除的一端称作 栈顶 ,另外一端称作 栈底 。 栈不支持随机访问 ,栈的数据元素遵循 后进先出 的原则,即 LIFO(Late In First Out)。 也许有人曾经听说过 压栈 和 入栈 的术语,以

    2024年02月03日
    浏览(34)
  • 【数据结构】庖丁解牛,图文结合带你轻松上手带头循环链表

    君兮_的个人主页 勤时当勉励 岁月不待人 C/C++ 游戏开发 Hello,米娜桑们,这里是君兮_,我们今天接着上回的单链表来讲讲带头双向循环链表,这种链表也是我们在实际应用中最常用的几种链表之一,学好这种链表是是非常重要的,我会尽量用通俗易懂的文字配合逻辑图来帮助

    2024年02月14日
    浏览(33)
  • 深入浅出带你玩转栈与队列——【数据结构】

    W...Y的主页 😊 代码仓库分享 💕 目录 1.栈 1.1栈的概念及结构 1.2栈的结构特征图  ​编辑 1.3栈的实现 1.3.1栈的初始化 1.3.2进栈 1.3.3出栈 1.3.4销毁内存 1.3.5判断栈是否为空 1.3.5栈底元素的读取 1.3.6栈中大小 1.4栈实现所有接口 2.队列 2.1队列的概念 2.2队列的结构   2.3队列的实

    2024年02月11日
    浏览(49)
  • 数据结构——栈和队列

    目录  一.前言 二.前文回顾 三.栈 3.1 栈的概念及结构 3.2 栈的实现 3.2.1 初始化函数 3.2.2 销毁函数 3.2.3 入栈函数 3.2.4 出栈函数 3.2.5 计算大小函数 3.2.6 空栈函数 3.2.7 获取栈顶函数  3.2.8 小测试 3.3 全部代码 四.栈的练习 4.1 有效的括号 五.队列 5.1队列的概念及结构 5.2 队列的实

    2024年01月25日
    浏览(33)
  • 数据结构 | 栈和队列

    … 📘📖📃本文已收录至:数据结构 | C语言 更多知识尽在此专栏中! 栈(Stack) 又名堆栈,它是一种运算受限的线性表,限定仅在表尾进行插入和删除操作的线性表。 队列(Queue) 也是一种特殊的线性表,特殊之处在于它只允许在表的前端(Front)进行删除操作,而在表的

    2024年01月23日
    浏览(34)
  • 数据结构--栈和队列

    栈是一种常见的数据结构,它遵循 后进先出LIFO (Last In First Out)的原则。 进行数据插入和操作的一端称为栈顶,另一端称为栈底 。 压栈 :栈的插入操作被称为压栈/进栈/入栈,入数据在栈顶。 出栈 :栈的删除操作。出数据也在栈顶; 栈可以用 数组 或者是 链表 来实现;

    2024年02月09日
    浏览(32)
  • 栈和队列【数据结构】

    2024年02月16日
    浏览(30)
  • 数据结构---栈和队列

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

    2024年01月18日
    浏览(31)
  • [数据结构】栈和队列

    目录 1.栈 1.1概念 1.2 栈的使用 1.3.栈的模拟实现 2.队列 2.1概念 2.2队列的使用 2.3队列的模拟实现 2.4 循环队列 2.5双端队列   栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素

    2024年02月07日
    浏览(30)
  • 数据结构【栈和队列】

    一、栈 1.定义:只允许一端进行插入和删除的线性表,结构与手枪的弹夹差不多,可以作为实现递归函数(调用和返回都是后进先出)调用的一种数据结构; 栈顶:允许插入删除的那端; 栈底:固定的,不允许插入或删除; 空栈:不含元素; 2.特点:后进先出; 3.操作:入

    2024年02月15日
    浏览(32)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包