【数据结构】线性表之《栈》超详细实现

  • 一.栈的概念及结构
  • 二.顺序栈与链栈
    • 1.顺序栈
    • 2.链栈
      • 1.单链表栈
      • 2.双链表栈
  • 三.顺序栈的实现
    • 1.栈的初始化
    • 2.检查栈的容量
    • 3.入栈
    • 4.出栈
    • 5.获取栈顶元素
    • 6.栈的大小
    • 7.栈的判空
    • 8.栈的清空
    • 9.栈的销毁
  • 四.模块化源代码
    • 1.Stack.h
    • 2.Stack.c
    • 3.test.c

一.栈的概念及结构

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

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

二.顺序栈与链栈

栈的实现一般可以使用数组或者链表实现,相对而言数组的结构实现更优一些。因为数组在尾上
插入数据的代价比较小。那是为什么?且听下文分解。

1.顺序栈

在这里插入图片描述

2.链栈

1.单链表栈

在这里插入图片描述

将栈顶与栈低换个位置可以解决该问题,如下图:

在这里插入图片描述

2.双链表栈

在这里插入图片描述

  1. 由于双向链表比单链表多一个指针,基于节省内存的原由单链表优于双向链表
  2. 数组的效率更优于单链表,原因:链表每一次插入一个数据都要申请一个节点,每次删除一个数据都要释放一个节点,且顺序栈包含数据+容量+栈顶,而链栈包含数据+指针,每个数据都要包含指针,顺序栈较于连栈会省一些内存。

接下来我将实现最优的——>顺序栈

三.顺序栈的实现

会写顺序表,那么实现顺序栈会非常轻松,这里就不一一介绍了,直接上代码。

1.栈的初始化

void StackInit(ST* ps)
{assert(ps);//断言ps->arr = NULL;ps->capacity = 0;ps->top = 0;
}

2.检查栈的容量

void CheckCapacity(ST* ps)
{assert(ps);//栈满if (ps->top == ps->capacity){int newCapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;STDataType* tmp = (STDataType*)realloc(ps->arr, sizeof(STDataType) * newCapacity);if (tmp == NULL)//空间开辟失败{perror("realloc fail!");exit(-1);//退出程序}//空间开辟成功ps->arr = tmp;ps->capacity = newCapacity;}
}

3.入栈

void StackPush(ST* ps, STDataType x)
{assert(ps);CheckCapacity(ps);ps->arr[ps->top] = x;ps->top++;
}

4.出栈

void StackPop(ST* ps)
{assert(ps);assert(ps->top > 0);//栈空,无法出栈,否则下标越界ps->top--;
}

5.获取栈顶元素

STDataType StackTop(ST* ps)
{assert(ps);assert(ps->top > 0);return ps->arr[ps->top - 1];
}

6.栈的大小

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

7.栈的判空

bool StackEmpty(ST* ps)
{assert(ps);return ps->top == 0;
}

8.栈的清空

void StackClear(ST* ps)
{assert(ps);ps->top = 0;ps->capacity = 0;
}

9.栈的销毁

void StackDestory(ST* ps)
{assert(ps);StackClear(ps);free(ps->arr);ps->arr = NULL;
}

四.模块化源代码

1.Stack.h

//#pragma once 防止头文件被重复包含
#ifndef __STACK_H__
#define __STACK_H__#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>typedef int STDataType;typedef struct Stack
{STDataType* arr; //栈空间的首地址int top;         //栈顶int capacity;    //容量
}ST;void StackInit(ST* ps);//栈的初始化void CheckCapacity(ST* ps);//检查栈的容量void StackPush(ST* ps, STDataType x);//入栈void StackPop(ST* ps);//出栈STDataType StackTop(ST* ps);//获取栈顶元素int StackSize(ST* ps);//获取栈的长度bool StackEmpty(ST* ps);//栈的判空void StackClear(ST* ps);//栈的清空void StackDestory(ST* ps);//栈的销毁#endif

2.Stack.c

#define _CRT_SECURE_NO_WARNINGS 1#include"Stack.h"void StackInit(ST* ps)
{assert(ps);//断言ps->arr = NULL;ps->capacity = 0;ps->top = 0;
}void CheckCapacity(ST* ps)
{assert(ps);//栈满if (ps->top == ps->capacity){int newCapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;STDataType* tmp = (STDataType*)realloc(ps->arr, sizeof(STDataType) * newCapacity);if (tmp == NULL)//空间开辟失败{perror("realloc fail!");exit(-1);//退出程序}//空间开辟成功ps->arr = tmp;ps->capacity = newCapacity;}
}void StackPush(ST* ps, STDataType x)
{assert(ps);CheckCapacity(ps);ps->arr[ps->top] = x;ps->top++;
}void StackPop(ST* ps)
{assert(ps);assert(ps->top > 0);//栈空,无法出栈,否则下标越界ps->top--;
}STDataType StackTop(ST* ps)
{assert(ps);assert(ps->top > 0);return ps->arr[ps->top - 1];
}int StackSize(ST* ps)
{assert(ps);return ps->top;
}bool StackEmpty(ST* ps)
{assert(ps);return ps->top == 0;
}void StackClear(ST* ps)
{assert(ps);ps->top = 0;ps->capacity = 0;
}void StackDestory(ST* ps)
{assert(ps);StackClear(ps);free(ps->arr);ps->arr = NULL;
}

3.test.c

#define _CRT_SECURE_NO_WARNINGS 1#include"Stack.h"enum
{EXIT,PUSH,POP,TOP,SIZE,EMPTY,CLEAR
};void Menu()
{printf("***************栈**************\n");printf("****1.入栈           2.出栈****\n");printf("****3.栈顶           4.大小****\n");printf("****5.判空           6.清空****\n");printf("****0.退出               ******\n");printf("*******************************\n");
}int main()
{ST st;StackInit(&st);int select = 0;STDataType value;bool flag;do{Menu();printf("请选择您的操作:");scanf("%d", &select);switch (select){case EXIT:printf("退出栈!\n");break;case PUSH:printf("请输入您要入栈的值:");scanf("%d", &value);StackPush(&st, value);break;case POP:StackPop(&st);break;case TOP:value = StackTop(&st);printf("栈顶元素:%d\n", value);break;case SIZE:printf("栈的大小为:%d\n", StackSize(&st));break;case EMPTY:flag = StackEmpty(&st);if (flag){printf("栈为空!\n");}else{printf("栈不为空!\n");}break;case CLEAR:StackClear(&st);break;default:printf("输入错误,请重新输入!\n");break;}} while (select);StackDestory(&st);return 0;
}

创作不易,如果能帮到你的话能赏个三连吗?感谢啦!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/32195.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

程序猿成长之路之数据挖掘篇——决策树分类算法(1)——信息熵和信息增益

决策树不仅在人工智能领域发挥着他的作用&#xff0c;而且在数据挖掘中也在分类领域中独占鳌头。了解决策树的思想是学习数据挖掘中的分类算法的关键&#xff0c;也是学习分类算法的基础。 什么是决策树 用术语来说&#xff0c;决策树&#xff08;Decision Tree&#xff09;是…

Go自定义数据的序列化流程

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

数据库设计概述-数据库设计内容、数据库设计方法(基于E-R模型的规范设计方法)

一、引言 如何利用关系数据库理论设计一个满足应用系统需求的数据库 二、数据库设计内容 1、数据库设计是基于应用系统需求分析中对数据的需求&#xff0c;解决数据的抽象、数据的表达和数据的存储结构等问题 2、其目标是设计出一个满足应用要求、简洁、高效、规范合理的数…

Map集合之HashMap细说

最近在看面试题&#xff0c;看到了hashmap相关的知识&#xff0c;面试中问的也挺多的&#xff0c;然后我这里记录下来&#xff0c;供大家学习。 Hashmap为什么线程不安全 jdk 1.7中&#xff0c;在扩容的时候因为使用头插法导致链表需要倒转&#xff0c;从而可能出现循环链表问…

航行在水域:使用数据湖构建生产级 RAG 应用程序

在 2024 年年中&#xff0c;创建一个令人印象深刻和兴奋的 AI 演示可能很容易。需要一个强大的开发人员&#xff0c;一些聪明的提示实验&#xff0c;以及一些对强大基础模型的API调用&#xff0c;你通常可以在一个下午建立一个定制的AI机器人。添加一个像 langchain 或 llamain…

c++ 内存分析模型、引用

一、内存模型分区 内存四区的意义&#xff1a; 不同区域存放的数据&#xff0c;赋予不同的生命周期&#xff0c;给我们更大的灵活编程 &#xff08;一&#xff09;程序运行前 在程序编译后&#xff0c;生成了exe可执行程序&#xff0c;未执行程序前分为两个区域 代码区&…

2024hw蓝队面试题--4

SQL注入特征&#xff0c;误报原因以及怎么处理告警&#xff1f; 1.非法字符检测&#xff1a;这种类型的攻击通常会使用特殊字符&#xff0c;如单引号()、双引号(")、分号(;)、注释符号(--或/、/)等。检测输入中是否包含这类特殊字符是识别SQL注入攻击的一种方法。 2.堆栈…

PostgreSQL源码分析——审计插件pgaudit

PostgreSQL审计插件pgaudit 在PostgreSQL中&#xff0c;提供了开源的审计插件pgaudit&#xff0c;但是其功能并不完善&#xff0c;只提供了基本的审计功能&#xff0c;对此&#xff0c;很多基于PG开发的商业数据库大多提供了丰富的审计功能。比如人大金仓&#xff0c;openGaus…

健康实训室:老年保健与管理实训室的建设方案

一、建设背景和意义 我国正处于人口老龄化的加速期,随着老龄人口的不断增加,老年人的健康与养老问题已成为社会关注的热点问题。针对这一现状,建设老年保健与管理实训室具有重要的现实意义和战略价值: 1、培养高素质的老年保健和管理人才。老年保健与管理实训室的建设可以为医…

IOS Swift : 从入门到精通结构、属性和方法 结构体,第一部分

文章目录 创建自己的结构计算属性属性观察者方法变异方法字符串的属性和方法数组的属性和方法 创建自己的结构 Swift 允许你以两种方式设计自己的类型&#xff0c;其中最常见的是结构&#xff0c;或简称为structs。结构可以拥有自己的变量和常量&#xff0c;以及自己的函数&am…

浅谈Java23种设计模式之结构型模式的几种使用场景

前言 这是设计模式的第二期;继续根据实际开发应用场景解析这几种结构型设计模式. 1.适配器模式&#xff08;Adapter&#xff09; 概念: 它允许两个不兼容的接口通过适配器类工作在一起。这种模式通常用于将已存在的类&#xff08;被称为适配者&#xff09;的接口转换成客户端…

升级指南:探索CMMI2.0与3.0之间的企业变革!

CMMI2.0和CMMI3.0对企业的要求在某些方面有所变化&#xff0c;主要体现在以下几个方面&#xff1a; CMMI2.0对企业的要求 1.人员要求&#xff1a; 硬性要求&#xff1a;确保企业有25名以上的技术人员和10名以上的支持人员。 设立专门的人员对接CMMI评估&#xff0c;负责体系…

SpringMVC系列七: 手动实现SpringMVC底层机制-上

手动实现SpringMVC底层机制 博客的技术栈分析 &#x1f6e0;️具体实现细节总结 &#x1f41f;准备工作&#x1f34d;搭建SpringMVC底层机制开发环境 实现任务阶段一&#x1f34d;开发ZzwDispatcherServlet&#x1f966;说明: 编写ZzwDispatcherServlet充当原生的DispatcherSer…

码云建仓库

1.新建仓库 码云地址 打开 码云地址 &#xff0c;点击“”&#xff0c;新建仓库&#xff0c;添加仓库内容 &#xff0c;创建。 小提示&#xff1a;如果本地已有项目&#xff0c;就不要选初始化&#xff0c;设置模板&#xff0c;容易冲突。 2. 进入当前仓库页 小提示&#x…

【2024最新华为OD-C/D卷试题汇总】[支持在线评测] LYA的巡演(100分) - 三语言AC题解(Python/Java/Cpp)

&#x1f36d; 大家好这里是清隆学长 &#xff0c;一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-C/D卷的三语言AC题解 &#x1f4bb; ACM银牌&#x1f948;| 多次AK大厂笔试 &#xff5c; 编程一对一辅导 &#x1f44f; 感谢大家的订阅➕ 和 喜欢&#x1f497; &#x1f…

心明眼亮 洞悉万物

如何洞悉事物的本质呢&#xff1f; 阳明先生&#xff1a;世间之事&#xff0c;纷繁复杂&#xff0c;不可能一一研究得过来。 圣人只需要把内心的明镜擦亮&#xff0c;而无需担心外部的事事物物在镜子中如何映照。 —— 外界事物是无穷无尽的&#xff0c;永远探究不完&#xf…

30 - 每位经理的下属员工数量(高频 SQL 50 题基础版)

30 - 每位经理的下属员工数量 -- 根据reports_to &#xff0c;获取employee_id,即分组用e1.reports_to&#xff0c;查询用e2.employee_id,e2.nameselect e2.employee_id,e2.name ,count(e1.reports_to) reports_count,round(avg(e1.age),0) average_age from Employees e1 left…

【鸿蒙踩坑记录】解决:list组件滑动至左边或右边,回弹效果过大问题

一、问题描述 开发过程中使用List组件&#xff0c;当内容超过一屏时可出现滚动效果&#xff0c;此时按住内容迅速滑动至左边&#xff0c;或者滑动到右边&#xff0c;回弹效果过大 期望&#xff1a;滑动时&#xff0c;不要有那么大的回弹效果 二、目前效果 三、解决方法 3.1…

go语言day03

目录 一、 go语言的数据类型&#xff1a; 二、声明赋值的简写形式&#xff1a; ":" 1&#xff09;重复使用的编译错误 2&#xff09;在全局变量中使用 : 会报编译错误 三、变量规则&#xff1a; 0&#xff09;变量的命名规则&#xff1a; 1&#xff09;创建的局部…

丰臣秀吉-读书笔记六

登山的目标必然是山顶。但人生的乐趣和生息的快乐却不在山顶&#xff0c;相反可以说是在山中的逆境之处。当我们遇上峡谷、绝壁、溪流、断崖、雪崩之类的险路时&#xff0c;心里虽想着已经不行了等&#xff0c;却不甘就此罢手而不与面前的艰难险阻战斗。而当我们完美克服并跨越…