数据结构深入理解--栈

简介: 数据结构深入理解--栈

一、栈的定义

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

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

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

       

       栈可以这样理解:相信大家都对枪械有一定粗略的了解,咱们就用压子弹来帮助大家进行理解。

       压栈,大家可以想象为压子弹,子弹是一发一发往下压,那压栈就是在容量之内一个数据一个数据往下压。

       出栈,大家可以理解为子弹射出的过程,即:最后压入的子弹先出。数据最后的先出。

       这就是压栈与出栈的过程,当然也可以压一个出一个,压多个出一个都可以,大家完全就可以把栈当作压子弹和子弹射出。

       好了,基础知识我们已经掌握,那么,我们该用什么结构来实现栈呢?

       通过单链表,双链表还是什么?大家可以在此处进行思考,稍后公布答案。

二、栈的实现

       上文说到,我们要选择一个结构来实现栈。我们来一一分析一下:

       双链表全称为:带头双向循环链表,用它来实现可以吗?还用说吗?太过于完美当然可以,但是要用两个指针,同学们,一个指针已经困扰大家已久,那两个指针想必是大家不想经历的大恐怖。所以,这个时候咱们把它先列为备胎(实在没办法在想它(在特殊情况下能渣则渣)🐶)。

       单链表全称为:不带头单向不循环链表,大家想单链表找到尾元素麻烦吗?找一遍时间复杂度为O(N)。虽然我们可以反转一下,但是你愿意用吗?要是放两个指针还不如用双链表。

       这个时候怎么办?难道我们要使用双链表?不,绝对不行。这时,数组意外路过,对啊,我们可以用数组。

       数组每次使用前像顺序表一样判断是否开辟空间,在用一个变量size来记录尾,这样不就完美符合要求了。那说干就干吧。打开我们心爱的VS。

       2.1 栈的结构

               栈的结构,可以借鉴一下顺序表,要有数组、容量和栈定元素。结构如下:

1. // 支持动态增长的栈
2. typedef int STDataType;
3. typedef struct Stack
4. {
5.  STDataType* a;
6.  int top;    // 栈顶
7.  int capacity;  // 容量 
8. }Stack;

       2.2 栈的初始化

               要进行初始化,大家想一想top赋值为多大合适,如果为0,那么0是不是为栈的第一个元素?是不是?答案是:是的。那么,有没有办法不叫top指向数组第一个元素?有,使top的值为-1即可。在后续代码中,我会将top初始化为0(别问,问就是top此时可以当顺序表中的size使用)。代码如下:

1. // 初始化栈
2. void StackInit(Stack* ps)
3. {
4.  assert(ps);
5.  ps->a = NULL;
6.  ps->capacity = ps->top = 0;
7. }

       2.3 栈的销毁

               在我们今后写代码一定要记住:只要你malloc,realloc一定要free,你创建了就一定要销毁。

               那我们创建了一个栈,那么我们一定要销毁。代码如下:

1. // 销毁栈 
2. void StackDestroy(Stack* ps)
3. {
4.  assert(ps);
5.  free(ps->a);//此处记得释放ps指向的数组,不要写成ps!!!
6.  ps->a = NULL;
7.  ps->capacity = ps->top = 0;
8. }

               注意事项写在代码里了,一定要记住!!!

       2.3 栈元素的插入

1. void StackPush(Stack* ps, STDataType data)
2. {
3.  assert(ps);
4.  if (ps->capacity == ps->top)
5.  {
6.    int newcapacity = ps->capacity == 0 ? 4 : 2 * sizeof(ps->capacity);
7.    STDataType* newnode = (STDataType*)realloc(ps->a,newcapacity*sizeof(STDataType));
8.    if (newnode == NULL)
9.    {
10.       perror("realloc fail");
11.       return;
12.     }
13.     ps->capacity = newcapacity;
14.     ps->a = newnode;
15.   }
16.   //这里之所以没有封装成一个接口,是因为这里只用一次,其余的都不使用
17.   ps->a[ps->top] = data;
18.   ps->top++;
19.   //这里也可以合二为一
20.   //ps->a[ps->top++] = data;
21. }

               此处要点与顺序表类似,就不过多强调。

       2.4 栈元素的删除

1. // 出栈 
2. void StackPop(Stack* ps)
3. {
4.  assert(ps);
5.  assert(ps->top > 0);
6.  ps->top--;
7. }

               此处代码过于简单,那么能不能不把这个封装了,直接写。其实你要是想这么干,你可以试一试,不过提醒一下:可能会出乱子。还是那句话:专业的事专业的人做。

       2.5 栈顶元素获取

1. // 获取栈顶元素 
2. STDataType StackTop(Stack* ps)
3. {
4.  assert(ps);
5.  assert(ps->top > 0);
6.  return ps->a[ps->top - 1];
7. }

               对于此代码最后的返回值可能会有人有疑问,我简单解释一下:  

       2.6 栈元素有效个数获取

1. // 获取栈中有效元素个数 
2. int StackSize(Stack* ps)
3. {
4.  assert(ps);
5.  return ps->top;
6. }

               上文说过top初始化为0时可当size来使用,代码简单,不过多解释。

       2.7 栈是否为空判断

1. // 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
2. bool StackEmpty(Stack* ps)
3. {
4.  assert(ps);
5.  return ps->top == 0;
6. }

               注意点:必须包含头文件:stdbool.h。

三、代码总览

       Stack.h

1. #pragma once
2. #include<stdio.h>
3. #include<stdlib.h>
4. #include<assert.h>
5. #include<stdbool.h>
6. 
7. // 支持动态增长的栈
8. typedef int STDataType;
9. typedef struct Stack
10. {
11.   STDataType* a;
12.   int top;    // 栈顶
13.   int capacity;  // 容量 
14. }Stack;
15. 
16. // 初始化栈 
17. void StackInit(Stack* ps);
18. // 入栈 
19. void StackPush(Stack* ps, STDataType data);
20. // 出栈 
21. void StackPop(Stack* ps);
22. // 获取栈顶元素 
23. STDataType StackTop(Stack* ps);
24. // 获取栈中有效元素个数 
25. int StackSize(Stack* ps);
26. // 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
27. bool StackEmpty(Stack* ps);
28. // 销毁栈 
29. void StackDestroy(Stack* ps);

       Stack.c

1. #include"Stack.h"
2. 
3. // 初始化栈
4. void StackInit(Stack* ps)
5. {
6.  assert(ps);
7.  ps->a = NULL;
8.  ps->capacity = ps->top = 0;
9. }
10. // 入栈 
11. void StackPush(Stack* ps, STDataType data)
12. {
13.   assert(ps);
14.   if (ps->capacity == ps->top)
15.   {
16.     int newcapacity = ps->capacity == 0 ? 4 : 2 * sizeof(ps->capacity);
17.     STDataType* newnode = (STDataType*)realloc(ps->a,newcapacity*sizeof(STDataType));
18.     if (newnode == NULL)
19.     {
20.       perror("realloc fail");
21.       return;
22.     }
23.     ps->capacity = newcapacity;
24.     ps->a = newnode;
25.   }
26.   //这里之所以没有封装成一个接口,是因为这里只用一次,其余的都不使用
27.   ps->a[ps->top] = data;
28.   ps->top++;
29.   //这里也可以合二为一
30.   //ps->a[ps->top++] = data;
31. }
32. // 出栈 
33. void StackPop(Stack* ps)
34. {
35.   assert(ps);
36.   assert(ps->top > 0);
37.   ps->top--;
38. }
39. // 获取栈顶元素 
40. STDataType StackTop(Stack* ps)
41. {
42.   assert(ps);
43.   assert(ps->top > 0);
44.   return ps->a[ps->top - 1];
45. }
46. // 获取栈中有效元素个数 
47. int StackSize(Stack* ps)
48. {
49.   assert(ps);
50.   return ps->top;
51. }
52. // 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
53. bool StackEmpty(Stack* ps)
54. {
55.   assert(ps);
56.   return ps->top == 0;
57. }
58. // 销毁栈 
59. void StackDestroy(Stack* ps)
60. {
61.   assert(ps);
62.   free(ps->a);//此处记得释放ps指向的数组,不要写成ps!!!
63.   ps->a = NULL;
64.   ps->capacity = ps->top = 0;
65. }

       测试代码:test.c

1. #include"Stack.h"
2. 
3. int main()
4. {
5.  Stack p;
6.  StackInit(&p);
7.  StackPush(&p, 1);
8.  StackPush(&p, 2);
9.  StackPush(&p, 3);
10.   StackPush(&p, 4);
11.   while (!StackEmpty(&p))
12.   {
13.     printf("%d ", StackTop(&p));
14.     StackPop(&p);
15.   }
16.   StackDestroy(&p);
17.   return 0;
18. }

四、例题

       既然明白了,那么来几道题巩固一下吧!

       例一:

       设栈S和队列 Q的初始状态均为空,元素 abcdepg 依次进入栈S。若每个元素出栈后立即进入队列 Q,且7个元素出队的顺序是 bdcfeag,则栈S的容量至少是:

       

       例二:

      若元素a,b,c,d,e,f依次进栈,允许进栈、退栈操作交替进行,但不允许连续3次进行退栈操作,不可能得到的出栈序列是()。

       A. dcebfa                        B. cbdaef                        C.bcaefd                        D.afedcb

       例三:

       元素 a,b,c,d,e依次进入初始为空的栈中,若元素进栈后可停留、可出栈,直到所有元素都出栈,则在所有可能的出栈序列中,以元素d开头的序列个数是

       好了,我们的学习到现在就结束了,如有疑惑可私信,也可在评论区留言。

完!

相关文章
|
15天前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
90 9
|
6天前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
15 1
|
9天前
|
存储 算法 Java
数据结构的栈
栈作为一种简单而高效的数据结构,在计算机科学和软件开发中有着广泛的应用。通过合理地使用栈,可以有效地解决许多与数据存储和操作相关的问题。
|
12天前
|
存储 JavaScript 前端开发
执行上下文和执行栈
执行上下文是JavaScript运行代码时的环境,每个执行上下文都有自己的变量对象、作用域链和this值。执行栈用于管理函数调用,每当调用一个函数,就会在栈中添加一个新的执行上下文。
|
14天前
|
存储
系统调用处理程序在内核栈中保存了哪些上下文信息?
【10月更文挑战第29天】系统调用处理程序在内核栈中保存的这些上下文信息对于保证系统调用的正确执行和用户程序的正常恢复至关重要。通过准确地保存和恢复这些信息,操作系统能够实现用户模式和内核模式之间的无缝切换,为用户程序提供稳定、可靠的系统服务。
41 4
|
1月前
|
算法 程序员 索引
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
栈的基本概念、应用场景以及如何使用数组和单链表模拟栈,并展示了如何利用栈和中缀表达式实现一个综合计算器。
30 1
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
|
18天前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
1月前
初步认识栈和队列
初步认识栈和队列
58 10
|
1月前
数据结构(栈与列队)
数据结构(栈与列队)
17 1
|
1月前
|
算法
数据结构与算法二:栈、前缀、中缀、后缀表达式、中缀表达式转换为后缀表达式
这篇文章讲解了栈的基本概念及其应用,并详细介绍了中缀表达式转换为后缀表达式的算法和实现步骤。
44 3

热门文章

最新文章