首页 > 其他分享 >【线性表】之栈(C语言)

【线性表】之栈(C语言)

时间:2022-11-17 21:31:52浏览次数:50  
标签:ps 线性表 top 之栈 栈顶 C语言 assert Stack arry


  • ​​回顾​​
  • ​​栈​​
  • ​​结构定义​​
  • ​​初始化​​
  • ​​销毁​​
  • ​​入栈​​
  • ​​出栈​​
  • ​​返回栈顶元素​​
  • ​​返回栈中元素个数​​
  • ​​判断栈是否为空​​
  • ​​调用​​

回顾

顺序表和链表的区别和联系

顺序表:

优点:空间连续支持随机访问。

缺点:1.中间或前面的插入删除时间复杂度O(N)。

2.增容的代价比较大

链表(带头双向循环):

缺点:

以借点为单位存储,不支持随机访问。

优点:

1.任意位置插入删除时间复杂度为O(1)

2.没有增容消耗,按需申请结点空间,不用了直接释放。


栈也是线性表,在逻辑上还是挨着放的。

栈的概念以及结构

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

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

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

【线性表】之栈(C语言)_栈

实现方式:

  1. 数组实现

【线性表】之栈(C语言)_顺序表_02

总结:

相当于之前顺序表的尾插尾删,用尾做栈顶,非常合适,唯一缺陷就是,空间不够需要增容(影响不大)。

(顺序表——​​【线性表】之顺序表_半生瓜のblog-CSDN博客​​)

  1. 链表实现

【线性表】之栈(C语言)_数据_03

出数据得找到前一个,这样的话用双向链表更好一些。

(所以说数据结构并没有规定用什么方法实现,只要能实现就行,对比的就是效率而已。)

也可以将单链表反过来。

【线性表】之栈(C语言)_数据结构_04

总结:

如果用尾插做栈顶,用双向链表更好。

如果用单链表实现,就用头去做栈顶,这样入栈和出栈效率都是O(1)。

整体来说数组的效率更优一些。


结构定义

typedef int StackDataType;
typedef struct Stack
{
StackDataType* arry;
int top;//指向栈顶
int capacity;//栈的容量——能放几个数据
}Stack;

初始化

如果初识的top给0,意味着top指向栈顶的元素的下一个,top给-1,top指向栈顶元素。

一定不能为空的东西,可以使用断言来处理。OJ题不可以使用断言。

void StackInit(Stack* ps)
{
assert(ps);
ps->arry = (StackDataType*)malloc(sizeof(StackDataType)*4);
if (ps->arry == NULL)
{
printf("malloc fail");
exit(-1);
}
ps->capacity = 4;
ps->top = 0;
}

销毁

void StackDestory(Stack* ps)
{
assert(ps);
free(ps->arry);
ps->arry = NULL;
ps->top = ps->capacity =0 ;
}

入栈

void StackPush(Stack* ps, StackDataType x)
{
assert(ps);
//满了
if (ps->top == ps->capacity)
{
StackDataType* tmp = (StackDataType*)realloc(ps->arry, ps->capacity * 2 * sizeof(StackDataType));
if (tmp == NULL)
{
printf("realloc fail");
exit(-1);
}
else
{
ps->arry = tmp;
ps->capacity *= 2;
}
}
ps->arry[ps->top] = x;
ps->top++;
}

出栈

void StackPop(Stack* ps)
{
assert(ps);
//如果栈空了调用top,直接终止程序报错
assert(ps->top > 0);
ps->top--;
}

返回栈顶元素

StackDataType StackTop(Stack* ps)
{
assert(ps);
assert(ps->top > 0);
return ps->arry[ps->top - 1];
}

返回栈中元素个数

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

判断栈是否为空

bool StackEmpty(Stack* ps)
{
assert(ps);
return ps->top == 0;//真为空,假为非空。
}

小提示:

上面有的函数只有两行代码,如果直接用里面的那句代码,可以吗?
可以,但是不好,通过那句代码访问到,但严格来说你不应该去访问,这是一种耦合,耦合就是一种强关联,
调用函数,无需去想top在0还是在-1,只管用就完事了。(有点软件工程的思想)


调用

int main(void)
{
Stack ps;
StackInit(&ps);
StackPush(&ps,1);
StackPush(&ps,2);
StackPush(&ps,3);
while (!StackEmpty(&ps))
{
printf("%d ", StackTop(&ps));
//取完栈顶的数据,想取下一个,那就得删一下
StackPop(&ps);
}
printf("\n");
StackDestory(&ps);
return 0;
}


标签:ps,线性表,top,之栈,栈顶,C语言,assert,Stack,arry
From: https://blog.51cto.com/u_15333750/5866185

相关文章

  • 【线性表】之队列(C语言)
    队列​​队列的概念​​​​结构定义​​​​初始化​​​​销毁​​​​队尾入​​​​队头出​​​​队头出​​​​队头数据​​​​队尾数据​​​​是否为空​​​​返......
  • LeetCode刷题(7)【栈&队列】用队列实现栈(C语言)
    用队列实现栈225.用队列实现栈-力扣(LeetCode)(leetcode-cn.com)目的:用队列实现栈,从先进先出——>先进后出,1234这四个数据依次从队列1的队尾进入,要让4先出,一个队列是无法......
  • LeetCode刷题(2)【链表】【合链表&链表的中间结点】(C语言)
    我的小站——半生瓜のblog快慢指针问题:思路:定义一个快指针和一个慢指针,快指针走到结束的时候,慢指针刚好走到一半。链表的中间结点。876.链表的中间结点-力扣(LeetCode)(le......
  • LeetCode刷题(3)【链表】【环形链表】&扩展(C语言)
    我的小站——半生瓜のblog环形链表141.环形链表-力扣(LeetCode)(leetcode-cn.com)什么是链表带环:链表的最后一个元素不指向空而指向前面的某个结点。思路:快慢指针,慢指针走......
  • C语言二级错题积累(4)
    在栈中,栈项指针的动态变化决定栈中元素的个数。详细设计的人物是为软件结构体中的每一个模块确定实现算法和局部数据结构,用某种选定的表达工具表示算法和数据结构的细节。扇......
  • C语言二级错题积累(5)
    常用的连续存储管理技术有固定分区存储管理和可变分区存储管理。程序流程图中带有箭头的线段表示的是控制流。若二叉树没有叶子结点,则为空二叉树。带链栈的栈底指针是随栈的......
  • C语言实现贪吃蛇小程序
    参考视频https://www.bilibili.com/video/BV1LN41197zV?from=search&seid=15462998985727977257代码有点缺陷:1.食物有可能会生成在吃不到的地方2.吃掉食物的音效添加失败//......
  • C语言实现推箱子小游戏
    C语言实现推箱子小游戏包括黑窗和图形界面参考视频https://www.bilibili.com/video/BV1By4y1a79o?t=4428BUG:当人进入到目的地的时候会无法移动。#include<stdio.h>#incl......
  • C语言自定义数据类型
    结构体参考视频:https://www.bilibili.com/video/BV1oi4y1g7CF?p=58大纲:结构体的声明结构体的自引用结构体内存对齐结构体传参结构体实现位段(位段的填充&可移植性)charshor......
  • C语言实现飞翔的小鸟小游戏
    参考视频https://www.bilibili.com/video/BV1Xo4y1R7hs缺陷:撞柱子功能暂未实现//飞翔的小鸟#include<stdio.h>//C语言标准头文件#include<graphics.h>//图形库头文件#includ......