【数据结构】之栈(C语言描述)
栈(Stack)是编程中最常用的数据结构之一。
栈的特点是“后进先出”,就像堆积木一样,堆的时候要一块一块堆到最上面,拆的时候需要从最上面一块一块往下拆。栈的原理也一样,只不过它的操作不叫堆和拆,而是叫入栈(或压栈)和出栈(或弹栈)。栈要求我们只能在栈顶(最上面的元素)处进行增加和删除。
栈可以用顺序表示法(顺序栈)和链式表示法(链栈)。由于栈只需要在一端增删节点,不需要在中间某处增删节点,因此即使栈在应用中大多数做的是增删操作,顺序栈虽然在增删操作时略微逊色于链栈(因为可能会扩容),但其存储密度要高于链栈,因此二者在性能方面可以说是各有各的好处。
下面的代码是使用 C语言 描述的链栈的代码。
栈的头文件 Stack.h 中的代码:
/**
* 栈(链栈):
* 本程序中栈的存储方式:栈顶->节点->节点->...->栈底。即:栈顶是第一个元素
*/
#include <Constant.h> // 栈中存储的数据的类型
typedef int ElemType; // 栈中的节点的数据结构体
typedef struct StackNode {
ElemType value; // 栈节点中存储的数据的值
struct StackNode* nextNode; // 下一个节点
} StackNode; // 栈的数据结构体
typedef struct Stack {
StackNode* data; // 栈中存储的所有节点
int length; // 栈的长度
StackNode* topNode; // 栈顶元素节点
} Stack; // 初始化一个空栈
Status initStack(Stack* S) {
S->data = (StackNode*)malloc(sizeof(StackNode));
if(S->data == NULL) {
printf("初始化栈失败!\n");
return FAILURE;
}
S->length = ;
S->topNode = NULL;
return SUCCESS;
} // 销毁栈
Status destroyStack(Stack* S) {
StackNode* tmpNode;
if(S->data == NULL) {
printf("栈不存在,销毁失败!\n");
return FAILURE;
}
while(S->topNode != NULL && S->topNode != S->data) {
tmpNode = S->topNode;
S->topNode = S->topNode->nextNode;
free(tmpNode);
}
S->data = NULL;
S->length = ;
S->topNode = NULL;
return SUCCESS;
} // 清空栈
Status clearStack(Stack* S) {
StackNode* tmpNode;
if(S->data == NULL) {
printf("栈不存在,清空失败!\n");
return FAILURE;
}
if(S->topNode == NULL) {
return SUCCESS;
}
while(S->topNode != S->data) {
tmpNode = S->topNode;
S->topNode = S->topNode->nextNode;
free(tmpNode);
}
S->topNode = NULL;
S->length = ;
return SUCCESS;
} // 判断栈是否为空
Status isStackEmpty(Stack* S) {
if(S->data == NULL) {
printf("栈不存在!\n");
exit();
}
if(S->length == ) {
return TRUE;
}
return FALSE;
} // 获取栈中元素的长度
int getStackLength(Stack* S) {
if(S->data == NULL) {
printf("栈不存在!\n");
exit();
}
return S->length;
} // 查看栈顶元素的值
ElemType getTopElem(Stack* S) {
if(S->data == NULL) {
printf("栈不存在,获取元素失败!\n");
exit();
}
if(S->topNode == NULL) {
printf("栈是空栈,获取元素失败!\n");
exit();
}
return S->topNode->value;
} // 元素入栈
Status push(Stack* S, ElemType e) {
StackNode* tmpNode;
if(S->data == NULL) {
printf("栈不存在,元素入栈失败!\n");
return FAILURE;
}
tmpNode = (StackNode*)malloc(sizeof(StackNode));
if(tmpNode == NULL) {
printf("元素入栈失败!\n");
return FAILURE;
}
tmpNode->value = e;
S->length++;
if(S->topNode == NULL) {
S->topNode = tmpNode;
S->topNode->nextNode = S->data;
return SUCCESS;
}
tmpNode->nextNode = S->topNode;
S->topNode = tmpNode;
return SUCCESS;
} // 元素出栈
StackNode* pop(Stack* S) {
StackNode* tmpNode;
if(S->data == NULL) {
printf("栈不存在,元素出栈失败!\n");
exit();
}
if(S->topNode == NULL) {
printf("栈是空栈,元素出栈失败!\n");
return NULL;
}
tmpNode = S->topNode;
S->topNode = S->topNode->nextNode;
if(S->topNode == S->data) {
S->topNode = NULL;
}
tmpNode->nextNode = NULL;
S->length--;
return tmpNode;
} // 遍历栈中的元素
void traverseStack(Stack* S) {
StackNode* tmpNode;
if(S->data == NULL) {
printf("栈不存在,遍历失败!\n");
exit();
}
if(S->topNode == NULL) {
printf("栈是空栈,遍历失败!\n");
exit();
}
printf("遍历栈:");
tmpNode = S->topNode;
while(tmpNode != S->data) {
printf("%-4d", tmpNode->value);
tmpNode = tmpNode->nextNode;
}
printf("\n");
} // 栈的测试函数
int testStack() {
// 数据声明
Stack stack;
StackNode* node;
int i;
// 初始化栈
if(initStack(&stack) == SUCCESS) {
printf("初始化栈成功!\n");
}
// 判断栈是否是空栈
printf("栈是否为空栈?%s\n", isStackEmpty(&stack) == TRUE ? "是" : "否");
// 元素入栈
for(i = ; i <= ; i++) {
if(push(&stack, i) == SUCCESS) {
printf("元素%d入栈成功!\n", i);
}
}
// 遍历栈中的元素
traverseStack(&stack);
// 元素出栈
node = pop(&stack);
if(node != NULL) {
printf("元素%d成功出栈!\n", node->value);
}
// 遍历栈中的元素
traverseStack(&stack);
// 查看栈顶元素的值
printf("栈顶元素的值是:%d\n", getTopElem(&stack));
// 获取栈的长度
printf("栈的当前长度:%d\n", getStackLength(&stack));
// 清空栈
if(clearStack(&stack) == SUCCESS) {
printf("清空栈成功!\n");
}
// 销毁栈
if(destroyStack(&stack) == SUCCESS) {
printf("销毁栈成功!\n");
}
return ;
}
常量类 Constant.h 中定义了一些常量,其代码如下:
#include <stdio.h>
#include <stdlib.h> #define TRUE 1
#define FALSE 0 #define SUCCESS 1
#define FAILURE 0 typedef int Status;
主函数所在的文件 main.c 中的代码如下:
#include <Stack.h>
int main() {
testStack();
return ;
}
运行结果如下:
初始化栈成功!
栈是否为空栈?是
元素1入栈成功!
元素2入栈成功!
元素3入栈成功!
元素4入栈成功!
元素5入栈成功!
遍历栈:5 4 3 2 1
元素5成功出栈!
遍历栈:4 3 2 1
栈顶元素的值是:4
栈的当前长度:4
清空栈成功!
销毁栈成功! Process returned 0 (0x0) execution time : 0.016 s
Press any key to continue.
【数据结构】之栈(C语言描述)的更多相关文章
- C语言学习书籍推荐《数据结构与算法分析:C语言描述(原书第2版)》下载
维斯 (作者), 冯舜玺 (译者) <数据结构与算法分析:C语言描述(原书第2版)>内容简介:书中详细介绍了当前流行的论题和新的变化,讨论了算法设计技巧,并在研究算法的性能.效率以及对运行 ...
- 数据结构与抽象 Java语言描述 第4版 pdf (内含标签)
数据结构与抽象 Java语言描述 第4版 目录 前言引言组织数据序言设计类P.1封装P.2说明方法P.2.1注释P.2.2前置条件和后置条件P.2.3断言P.3Java接口P.3.1写一个接口P.3. ...
- 数据结构与算法分析——C语言描述 第三章的单链表
数据结构与算法分析--C语言描述 第三章的单链表 很基础的东西.走一遍流程.有人说学编程最简单最笨的方法就是把书上的代码敲一遍.这个我是头文件是照抄的..c源文件自己实现. list.h typede ...
- 最小正子序列(序列之和最小,同时满足和值要最小)(数据结构与算法分析——C语言描述第二章习题2.12第二问)
#include "stdio.h" #include "stdlib.h" #define random(x) (rand()%x) void creat_a ...
- 《数据结构与算法分析——C语言描述》ADT实现(NO.00) : 链表(Linked-List)
开始学习数据结构,使用的教材是机械工业出版社的<数据结构与算法分析——C语言描述>,计划将书中的ADT用C语言实现一遍,记录于此.下面是第一个最简单的结构——链表. 链表(Linked-L ...
- 《数据结构与算法分析-Java语言描述》 分享下载
书籍信息 书名:<数据结构与算法分析-Java语言描述> 原作名:Data Structures and Algorithm Analysis in Java 作者: 韦斯 (Mark A ...
- 《数据结构与算法分析:C语言描述_原书第二版》CH3表、栈和队列_reading notes
表.栈和队列是最简单和最基本的三种数据结构.基本上,每一个有意义的程序都将明晰地至少使用一种这样的数据结构,比如栈在程序中总是要间接地用到,不管你在程序中是否做了声明. 本章学习重点: 理解抽象数据类 ...
- 《数据结构与算法分析——C语言描述》ADT实现(NO.01) : 栈(Stack)
这次的数据结构是一种特殊的线性表:栈(Stack) 栈的特点是后入先出(LIFO),可见的只有栈顶的一个元素. 栈在程序中的地位非常重要,其中最重要的应用就是函数的调用.每次函数调用时都会创建该函数的 ...
- 用链表实现栈----《数据结构与算法分析----C语言描述》
一.头文件: #ifndef _STACK_LINK_H_ #define _STACK_LINK_H_ struct stack_record; typedef struct stack_recor ...
- 读书笔记:《数据结构与算法分析Java语言描述》
目录 第 3 章 表.栈和队列 3.2 表 ADT 3.2.1 表的简单数组实现 3.2.2 简单链表 3.3 Java Collections API 中的表 3.3.1 Collection 接口 ...
随机推荐
- MyBatis的几个重要概念和工作流程
MyBatis 几个重要的概念 Mapper 配置: Mapper 配置可以使用基于 XML 的 Mapper 配置文件来实现,也可以使用基于 Java 注解的 MyBatis 注解来实现,甚至可以直 ...
- u检验粗浅理解
假设检验是以小概率事件,在一次实验中是不可能发生为前提(事实上是有可能发生的,但不是这样说的话,就落入一个圈,不能继续玩了),来否认原假设. u检验的定义: 已知从正态母体N(u,σ2)中抽得容量为n ...
- Zabbix 四 主动模式
本次的主机192.168.131.8 被动模式. 将zabbix4.4.4的源码包放过去,解压安装依赖准备编译安装,并创建zabbix账户. tar -xf zabbix-4.4.0.tar.gz & ...
- [考试反思]1024csp-s模拟测试85:以为
愈发垃圾. T1基本全场切(除了RP<-inf的zkt和把人擦) 然后T2想了半天逐渐趋近于正解,但是因为数据有问题锅了25分,没什么好说的.T3连题意转化都没有完成.括号匹配转为+1/-1做法 ...
- 七月月赛T2
题目描述 “X龙珠”是一款益智小游戏.游戏中有 n(2∣n) 个编号互不相同龙珠按照给定的顺序排成一个队列,每个龙珠上面都有一个编号.每次操作时,选择并取出龙珠队列中相邻的两个龙珠,放到目标队列的末尾 ...
- DEX文件解析---1、dex文件头解析
DEX文件解析---1.dex文件头解析 一.dex文件 dex文件是Android平台上可执行文件的一种文件类型.它的文件格式可以下面这张图概括: dex文件头一般固定为0x70个字 ...
- 网站搭建-虚拟机的使用-Linux (包括输入法和QQ下载使用)
之前已经联网了,基本的软件系统会自己下载,先不用管. 1. 先下载一个中文输入法吧: 先改一下Firefox的搜索引擎吧,因为大陆不支持google 下载,安装,就完事了,还好这个没变,几年不用这个系 ...
- 创建基于OData的Web API - Knowledge Builder API, Part I:Business Scenario
在.NET Core 刚刚1.0 RC的时候,我就给OData团队创建过Issue让他们支持ASP.NET Core,然而没有任何有意义的答复. Roadmap for ASP.NET Core 1. ...
- 后台服务器框架中的瑞士军刀——MCP
上篇介绍了一个简单的UDP服务框架,但是面对海量的请求,同步框架显然有点力不从心.于是在我接手好友系统的接口服务的时候,就采用了一个强大的异步框架——MCP框架. MCP框架是一个多进程异步框架,支持 ...
- 记录一次在Github写博客时的报错和解决方法
前几天刚刚搭建好了Github博客,打算用作记录Go语言学习笔记.由于在此前我没有使用过markdown语法写过博客,所以跟着文档了解了格式就想试试, 发表第一篇博客.markdown编辑器我用的是T ...