【数据结构之旅】顺序栈的定义、初始化、空栈判断、入栈、出栈操作

简介:

说明:

    往前学习数据结构,想运行一个完整的顺序栈的程序都运行不了,因为书上给的都是一部分一部分的算法,并没有提供一个完整可运行的程序,听了实验课,自己折腾了一下,总算可以写一个比较完整的顺序栈操作的小程序,对于栈也慢慢开始有了感觉。下面我会把整个程序拆开来做说明,只要把这些代码放在一个文件中,用编译器就可以直接编译运行了。





一、实现


1.程序功能

  关于栈操作的经典程序,首当要提及进制数转换的问题,利用栈的操作,就可以十分快速地完成数的进制转换。




2.预定义、头文件导入和类型别名

    代码如下:

1
2
3
4
5
6
7
8
9
10
# include <stdio.h>
# include <stdlib.h>
#define OVERFLOW - 1
#define ERROR  0
#define FALSE  0
#define TRUE  1
#define OK  1
 
typedef  int  ElemType;
typedef  int  Status;

    除了两个头文件的导入是必须的之外,下面做两点说明:

(1)其余的常量定义都是可选的,为的就是在下面的代码书写过程中可以尽量使用英文来表达程序的意思,而不是在代码的实现过程中直接使用数字,依个人喜欢,也可以直接使用数字;

(2)使用typedef做类型的别名也仅仅是为了程序中代码的意思更加清晰明了而已,实际也可以不这样使用;




3.顺序栈的定义

    代码如下:

1
2
3
4
5
6
typedef struct{
     ElemType *elem;      //存储空间的基址
     int  top;             //栈顶元素的下一个元素,简称栈顶位标
     int  size;            //当前分配的存储容量,作用看入栈操作就可以知道
     int  increment;       //扩容时,增加的存储容量,作用看入栈操作就可以知道
} SqStack;                   //顺序栈名称




4.栈的初始化

    代码如下:

1
2
3
4
5
6
7
8
Status InitStack_Sq(SqStack &S,  int  size,  int  inc){      //接受3个参数,&S是对结构体的引用
     S.elem = (ElemType*)malloc(size*sizeof(ElemType));   //分配存储空间
     if (S.elem == NULL)  return  OVERFLOW;      //判断上一步分配存储空间是否成功
     S.top =  0 ;             //置S为空栈,S.top为0即表示栈为空栈
     S.size = size;         //栈的空间初始容量值
     S.increment = inc;     //栈的空间初始增量值(如果需要扩容)
     return  OK;        //上面的执行正常,返回OK
}




5.空栈的判断

    代码如下:

1
2
3
4
5
6
7
8
Status StackEmpty_Sq(SqStack S){
     if (S.top ==  0 )
         return  TRUE;
     else
         return  FALSE;
}
//空栈的决断是,如果栈为空就返回1,否则就返回0,当然可以不这样规定;
//至于为什么要做空栈的判断,自然是有原因的,下面再看程序的代码时就可以知道了。




6.入栈

    代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
Status Push_Sq(SqStack &S, ElemType e){     //将元素e压入栈,这里e只是一个形参而已
     ElemType *newbase;         //定义中间变量
     if (S.top>= S.size){         //S.top如果指向最后一个不存储元素的地址时,即S.top大于
         newbase = (ElemType*)realloc(S.elem,  //等于S.size时,就表示栈满了
     (S.size + S.increment)*sizeof(ElemType));  //通过realloc动态扩容
     
     if (NULL == newbase)  return  OVERFLOW;  //判断扩容是否成功
     S.elem = newbase;       //扩容成功后才将中间变量的值指向S.elem,防止扩容失败时,
     S.size = S.size + S.increment;      //S.elem指向一个不是原来的位置
     }
     S.elem[S.top] = e;     //将e元素入栈
     S.top++;               //使S.top加1,表示指向的是栈顶位标
     return  OK;             //上面操作正常后返回1
}




7.出栈

    代码如下:

1
2
3
4
5
Status Pop_Sq(SqStack &S, ElemType &e){     //栈顶元素出栈,赋给元素e
     if ( 0  == S.top)  return  ERROR;    
     e = S.elem[--S.top];     //e出栈,并将S.top减1
     return  OK;
}




8.进制转换的函数

    其实上面的步骤操作都是为了创建一个顺序栈和定义顺序栈的操作而已,并对可能出现的各种情况做一些相应的举措,完毕后,下面就要使用上面创建的顺序栈以及栈的操作接口了,即在数制转换函数(这里是十进制转八进制)中使用上面的操作接口,代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
void  Converstion( int  N){
     SqStack S;
     ElemType e;
     InitStack_Sq(S,  10 5 );     //栈S的初始容量置为10,每次扩容容量为5
     
     while (N !=  0 ){
         Push_Sq(S, N% 8 );    //将N除以8的余数入栈
         N /=  8 ;             //N取值为其除以8的商
     }                           //理论基础为除8取余法
     
     while (StackEmpty_Sq(S) == FALSE){
         Pop_Sq(S, e);     //依次输出栈中的余数,并赋给元素e
         printf( "%d" , e);  //打印元素
     }




9.main函数

    进制转换函数调用栈操作的接口函数,以实现在数制转换过程中栈的操作;main函数调用数制转换函数,以实现数制的转换,代码如下:

1
2
3
4
5
int  main( void ){
     printf( "Enter a number:" );scanf( "%d" ,&num);
     Converstion(num);
     printf( "\n" );
}





二、执行

    有了上面的代码后,就可以在编译器中编译执行了,这里我是用c free 5来进行程序代码的编译:


(1)输入的数为1348时的结果:

wKiom1X-bHCSWO55AADAAgjs3ew463.jpg


(2)输入的数为2526时的结果:

wKioL1X-bsDTXhERAADDp63JNCE633.jpg





三、完整的代码

    下面把代码都放在一起:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
# include <stdio.h>
# include <stdlib.h>
#define OVERFLOW - 1
#define ERROR  0
#define FALSE  0
#define TRUE  1
#define OK  1
 
typedef  int  ElemType;
typedef  int  Status;
 
typedef struct{
     ElemType *elem;
     int  top;
     int  size;
     int  increment;
} SqStack;
 
Status InitStack_Sq(SqStack &S,  int  size,  int  inc){
     S.elem = (ElemType*)malloc(size*sizeof(ElemType));
     if (S.elem == NULL)  return  OVERFLOW;
     S.top =  0 ;
     S.size = size;
     S.increment = inc;
     return  OK;
}
 
Status StackEmpty_Sq(SqStack S){
     if (S.top ==  0 )
         return  TRUE;
     else
         return  FALSE;
}
 
Status Push_Sq(SqStack &S, ElemType e){
     ElemType *newbase;
     if (S.top>= S.size){
         newbase = (ElemType*)realloc(S.elem,
     (S.size + S.increment)*sizeof(ElemType));
     
     if (NULL == newbase)  return  OVERFLOW;
     S.elem = newbase;
     S.size = S.size + S.increment;
     }
     S.elem[S.top] = e;
     S.top++;
     return  OK;
}
 
Status Pop_Sq(SqStack &S, ElemType &e){
     if ( 0  == S.top)  return  ERROR;
     e = S.elem[--S.top];
     return  OK;
}
 
void  Converstion( int  N){
     SqStack S;
     ElemType e;
     InitStack_Sq(S,  10 5 );
     
     while (N !=  0 ){
         Push_Sq(S, N% 8 );
         N /=  8 ;
     }
     
     while (StackEmpty_Sq(S) == FALSE){
         Pop_Sq(S, e);
         printf( "%d" , e);
     }
}
 
int  main( void ){
     int  num;
     printf( "Enter a number:" );scanf( "%d" ,&num);
     Converstion(num);
     printf( "\n" );
}





本文转自 xpleaf 51CTO博客,原文链接:http://blog.51cto.com/xpleaf/1696524,如需转载请自行联系原作者
相关文章
|
2月前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
37 1
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
68 5
|
2月前
|
存储 算法 Java
数据结构的栈
栈作为一种简单而高效的数据结构,在计算机科学和软件开发中有着广泛的应用。通过合理地使用栈,可以有效地解决许多与数据存储和操作相关的问题。
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
53 0
|
2月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
230 9
|
2月前
|
存储 JavaScript 前端开发
执行上下文和执行栈
执行上下文是JavaScript运行代码时的环境,每个执行上下文都有自己的变量对象、作用域链和this值。执行栈用于管理函数调用,每当调用一个函数,就会在栈中添加一个新的执行上下文。
|
2月前
|
存储
系统调用处理程序在内核栈中保存了哪些上下文信息?
【10月更文挑战第29天】系统调用处理程序在内核栈中保存的这些上下文信息对于保证系统调用的正确执行和用户程序的正常恢复至关重要。通过准确地保存和恢复这些信息,操作系统能够实现用户模式和内核模式之间的无缝切换,为用户程序提供稳定、可靠的系统服务。
54 4
|
3月前
|
算法 程序员 索引
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
栈的基本概念、应用场景以及如何使用数组和单链表模拟栈,并展示了如何利用栈和中缀表达式实现一个综合计算器。
54 1
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
|
2月前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
3月前
初步认识栈和队列
初步认识栈和队列
65 10