C语言实现基础数据结构——栈

目录

栈的实现

数组栈

数组栈的实现

栈的初始化

栈的销毁

数据入栈

判断栈是否为空

数据出栈

获取栈顶元素

获取栈内数据个数

项目实现

栈的基础练习

有效的括号


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

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

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

📌

注意在栈中数据出栈的顺序不一定和入栈的顺序相同,当数据入栈后又出栈不影响下一个数据的入栈顺序

例如:

入栈1 2 3 4 5,若4和5是入栈又出栈的,则出栈的顺序应该是4 5 3 2 1

但是不会出现一种情况,即:

已经入栈的数据(未在入栈后立马出栈)不会出现在先入栈又立马出栈的数据的前方

例如:

入栈 1 2 3 4,若3和4是入栈又立马出栈,则不可能出现2和1在3和4的前面

2.若进栈序列为 1,2,3,4 ,进栈过程中可以出栈,则下列不可能的一个出栈序列是()
A 1, 4, 3, 2
B 2, 3, 4, 1
C 3, 1, 4, 2
D 3, 4, 2, 1
例如本题中,入栈顺序为1, 2, 3, 4,那么出栈顺序最基本的就是后入先出,即出栈顺序为4,3,2,1,如果入栈又立马出栈的是2,3,4,则有2,3,4,1,如果入栈又立马出栈的是1,则有1, 4, 3, 2,如果入栈又立马出栈的是3, 4,则有3, 4, 2, 1,但是不可能存在3, 1, 4, 2,因为1作为第一个入栈的,要么就是入栈又立马出栈,作为第一个出栈的,要么就是最后一个出栈的

栈的实现

栈的实现一般可以使用数组或者链表实现,相对而言数组的结构实现更优一些。因为栈满足后进先出或者即进即出,不需要额外移动数据,并且数组在尾上插入数据的代价比较小。

数组栈

//静态数组栈
typedef int STDataType;
#define N 10
typedef struct Stack
{STDataType _a[N];// 数组大小固定int _top; // 栈顶
}ST;//动态数组栈
typedef int STDataType;
typedef struct Stack
{STDataType* _a;// 数组大小不固定int _top;       // 栈顶int _capacity;  // 容量 
}ST;

数组栈的实现

//主要实现以下功能//栈的初始化
void STInit(ST* st);
//栈的销毁
void STDestroy(ST* st);
//数据入栈
void STPush(ST* st, STDataType x);
//数据出栈
void STPop(ST* st);
//判断栈是否为空
bool STEmpty(ST* st);
//获取栈元素
STDataType STTop(ST* st);
//获取栈数据个数
int STSize(ST* st);

栈的初始化

在初始化过程中注意top的初始值设置为0代表栈内数据的下一个位置,因为初始化栈代表栈内没有元素,如果用0代表栈内数据的当前位置,那么需要考虑该元素从何而来,既然没有数据,说明当前栈顶指针指向没有数据的位置,而数组下标为0代表第一个元素的位置,没有元素就不可能有第一个元素的位置,所以此时下标为0代表下一个元素的位置,此时说明栈内没有元素,但是准备在下一个位置(第一个元素)的位置添加数据

//栈的初始化
void STInit(ST* st)
{//判断是否存在队列assert(st);//初始化队列st->data = NULL;st->top = 0;//栈顶指针指向存储数据的下一个位置,代表栈内无数据//st->top = -1;//栈顶指针指向存储数据的位置,代表栈内无数据st->capacity = 0;
}

栈的销毁

//栈的销毁
void STDestroy(ST* st)
{//确保有栈的存在assert(st);//销毁栈free(st->data);st->data = NULL;//top和capacity更改为无数据的位置st->top = st->capacity = 0;
}

数据入栈

//数据入栈
void STPush(ST* st, STDataType x)
{//确保有栈的存在assert(st);//向top位置增加数据,并使top向后移动//需要判断栈的容量大小if (st->top == st->capacity){//如果栈的空间为0,则开辟四个空间,如果栈容量不为0,则扩容原来容量的2倍int newCapacity = st->capacity == 0 ? 4 : st->capacity * 2;STDataType* tmp = (STDataType*)realloc(st->data, sizeof(STDataType) * newCapacity);assert(tmp);st->data = tmp;//注意更新容量大小st->capacity = newCapacity;}//数据压栈并改变topst->data[st->top++] = x;
}

判断栈是否为空

//判断栈是否为空
bool STEmpty(ST* st)
{//确保有栈的存在assert(st);//栈为空返回真,栈不为空返回假return st->top == 0;//判断表达式返回值只有1和0,如果为真返回1(true),如果为假返回0(false)
}

数据出栈

//数据出栈
void STPop(ST* st)
{//确保有栈的存在assert(st);//确保栈不会越界assert(!STEmpty(st));//直接移动top指针,“看不见即删除”st->top--;
}

获取栈顶元素

//获取栈顶元素
STDataType STTop(ST* st)
{//确保栈存在assert(st);//确保栈不为空assert(!STEmpty(st));//top为栈内数据的下一个位置,要获取当前位置的元素需要-1操作return st->data[st->top - 1];
}

获取栈内数据个数

//获取栈内数据个数
int STSize(ST* st)
{assert(st);return st->top;
}

项目实现

//头文件
#pragma once#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>typedef int STDataType;
typedef struct stack
{STDataType* data;int top; //栈顶位置int capacity; //元素个数
}ST;//栈的初始化
void STInit(ST* st);
//栈的销毁
void STDestroy(ST* st);
//数据入栈
void STPush(ST* st, STDataType x);
//数据出栈
void STPop(ST* st);
//判断栈是否为空
bool STEmpty(ST* st);
//获取栈顶元素
STDataType STTop(ST* st);
//获取栈内数据个数
int STSize(ST* st);//实现文件
#include "stack.h"//栈的初始化
void STInit(ST* st)
{//判断是否存在队列assert(st);//初始化队列st->data = NULL;st->top = 0;//栈顶指针指向存储数据的下一个位置,代表栈内无数据//st->top = -1;//栈顶指针指向存储数据的位置,代表栈内无数据st->capacity = 0;
}//栈的销毁
void STDestroy(ST* st)
{//确保有栈的存在assert(st);//销毁栈free(st->data);st->data = NULL;//top和capacity更改为无数据的位置st->top = st->capacity = 0;
}//数据入栈
void STPush(ST* st, STDataType x)
{//确保有栈的存在assert(st);//向top位置增加数据,并使top向后移动//需要判断栈的容量大小if (st->top == st->capacity){//如果栈的空间为0,则开辟四个空间,如果栈容量不为0,则扩容原来容量的2倍int newCapacity = st->capacity == 0 ? 4 : st->capacity * 2;STDataType* tmp = (STDataType*)realloc(st->data, sizeof(STDataType) * newCapacity);assert(tmp);st->data = tmp;//注意更新容量大小st->capacity = newCapacity;}//数据压栈并改变topst->data[st->top++] = x;
}
//数据出栈
void STPop(ST* st)
{//确保有栈的存在assert(st);//确保栈不会越界assert(!STEmpty(st));//直接移动top指针,“看不见即删除”st->top--;
}
//判断栈是否为空
bool STEmpty(ST* st)
{//确保有栈的存在assert(st);//栈为空返回真,栈不为空返回假return st->top == 0;//判断表达式返回值只有1和0,如果为真返回1(true),如果为假返回0(false)
}
//获取栈顶元素
STDataType STTop(ST* st)
{//确保栈存在assert(st);//确保栈不为空assert(!STEmpty(st));//top为栈内数据的下一个位置,要获取当前位置的元素需要-1操作return st->data[st->top - 1];
}//获取栈内数据个数
int STSize(ST* st)
{assert(st);return st->top;
}//测试文件
#define _CRT_SECURE_NO_WARNINGS 1#include "stack.h"void test()
{ST st = { 0 };//初始化栈STInit(&st);//数据压栈STPush(&st, 1);STPush(&st, 2);STPush(&st, 3);STPush(&st, 4);STPush(&st, 5);//打印栈内数据while (!STEmpty(&st)){//打印栈顶数据printf("%d ", STTop(&st));//数据出栈以打印下一个数据STPop(&st);}//栈的销毁STDestroy(&st);
}int main()
{test();return 0;
}

栈的基础练习

有效的括号

题目链接:20. 有效的括号 - 力扣(LeetCode)

给定一个只包括 '('')''{''}''['']' 的字符串 s ,判断字符串是否有效。
有效字符串需满足:
左括号必须用相同类型的右括号闭合。
左括号必须以正确的顺序闭合。
每个右括号都有一个对应的相同类型的左括号。

思路解析:

本题可以使用栈来解决,当括号为左侧括号时,将括号存入栈内,当括号为右侧括号时,将该括号与栈顶的括号进行比较,比较一次进行一次出栈操作

📌

注意本题不一定要使用C语言来解决,因为C语言本身不支持泛型,需要单独先实现栈才能解题,本题使用C语言和栈的思路仅供参考

参考答案:

/** @lc app=leetcode.cn id=20 lang=c** [20] 有效的括号*/// @lc code=start
// 使用C语言和栈解决问题
// 栈的声明
typedef char STDataType;
typedef struct stack
{STDataType *data;int top;      // 栈顶位置int capacity; // 元素个数
} ST;// 栈的初始化
void STInit(ST *st);
// 栈的销毁
void STDestroy(ST *st);
// 数据入栈
void STPush(ST *st, STDataType x);
// 数据出栈
void STPop(ST *st);
// 判断栈是否为空
bool STEmpty(ST *st);
// 获取栈元素
STDataType STTop(ST *st);// 栈的实现
// 栈的初始化
void STInit(ST *st)
{// 判断是否存在队列assert(st);// 初始化队列st->data = NULL;st->top = 0; // 栈顶指针指向存储数据的下一个位置,代表栈内无数据// st->top = -1;//栈顶指针指向存储数据的位置,代表栈内无数据st->capacity = 0;
}// 栈的销毁
void STDestroy(ST *st)
{// 确保有栈的存在assert(st);// 销毁栈free(st->data);st->data = NULL;st->top = st->capacity = 0;
}// 数据入栈
void STPush(ST *st, STDataType x)
{// 确保有栈的存在assert(st);// 向top位置增加数据,并使top向后移动// 需要判断栈的容量大小if (st->top == st->capacity){// 如果栈的空间为0,则开辟四个空间,如果栈容量不为0,则扩容原来容量的2倍int newCapacity = st->capacity == 0 ? 4 : st->capacity * 2;STDataType *tmp = (STDataType *)realloc(st->data, sizeof(STDataType) * newCapacity);assert(tmp);st->data = tmp;st->capacity = newCapacity;}// 数据压栈并改变topst->data[st->top++] = x;
}
// 数据出栈
void STPop(ST *st)
{// 确保有栈的存在assert(st);// 确保栈不会越界assert(!STEmpty(st));// 直接移动top指针,“看不见即删除”st->top--;
}
// 判断栈是否为空
bool STEmpty(ST *st)
{// 确保有栈的存在assert(st);// 栈为空返回真,栈不为空返回假return st->top == 0; // 判断表达式返回值只有1和0,如果为真返回1(true),如果为假返回0(false)
}
// 获取栈元素
STDataType STTop(ST *st)
{// 确保栈存在assert(st);// 确保栈不为空assert(!STEmpty(st));// top为栈内数据的下一个位置,要获取当前位置的元素需要-1操作return st->data[st->top - 1];
}// 判断是否是有效括号
bool isValid(char *s)
{// 基本思路:左括号入栈,右括号时左括号出栈与右括号比较// 创建栈用于保存数据ST st = {0};// 初始化栈STInit(&st);// 数据入栈while (*s){if (*s == '(' || *s == '[' || *s == '{'){// 当是左括号时数据入栈STPush(&st, *s);}else{// 如果栈为空直接返回falseif (STEmpty(&st)){// 返回前释放空间,因为尽管没有数据,但是栈的空间已经开辟STDestroy(&st);return false;}// 当是右括号时数据出栈与右括号比较char top = STTop(&st);// 更新栈内元素STPop(&st);if ((*s == ')' && top != '(') || (*s == ']' && top != '[') || (*s == '}' && top != '{')){// 此处找不同,如果此处找相同,那么但凡出现一对满足的括号就会进入返回true//  返回之前销毁栈防止内存泄漏STDestroy(&st);return false;}}++s;}// 如果栈为空,则说明匹配完成,如果不为空说明栈内仍有括号没有匹配到,此时说明存在单独的括号bool ret = STEmpty(&st);// 返回之前销毁栈STDestroy(&st);return ret;
}
// @lc code=end

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

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

相关文章

后端设计PNR一点总结

条条大路通罗马 在追求极致PPA的过程中,时序问题总是可以解决 方法总比困难多 关键问题其实就是控制delay 不多不少,简单总结二十一条(欢迎大家评论区继续发挥): module padding的设置,可以有效解决congestion问题,factor自己try,命令:setPlaceMode -place_global…

代码随想录算法训练营Day40 | 343.整数拆分、96.不同的二叉搜索树

343.整数拆分 &#xff08;初见甚至看不出为什么这题是动规&#xff09; 本题大体思路是一个数可以被拆成两个或更多的数&#xff0c;而该数的最大乘积能由其拆成的数递推得到 注意点主要在于理解dp数组的定义 1、DP数组定义&#xff1a;一维数组&#xff0c;dp[i]表示 n i…

java.lang.IllegalStateException: Promise already completed.

spark submit 提交作业的时候提示Promise already complete 完整日志如下 File "/data5/hadoop/yarn/local/usercache/processuser/appcache/application_1706192609294_136972/container_e41_1706192609294_136972_02_000001/py4j-0.10.6-src.zip/py4j/protocol.py"…

Draw.io绘制UML图教程

一、draw.io介绍 1、draw.io简介 draw.io 是一款强大的免费在线图表绘制工具&#xff0c;支持创建流程图、组织结构图、时序图等多种图表类型。它提供丰富的形状库、强大的文本编辑和样式设置功能&#xff0c;使用户能够轻松创建专业级图表。draw.io 具有用户友好的界面&…

LeetCode 热题 100 | 二叉树(终)

目录 1 二叉树小结 1.1 模式一 1.2 模式二 2 236. 二叉树的最近公共祖先 3 124. 二叉树中的最大路径和 菜鸟做题&#xff08;返校版&#xff09;&#xff0c;语言是 C 1 二叉树小结 菜鸟碎碎念 通过对二叉树的练习&#xff0c;我对 “递归” 有了一些肤浅的理解。…

【激光SLAM】基于滤波的激光SLAM方法(Grid-based)

Filter-based SLAM 贝叶斯滤波数学概念贝叶斯滤波特性贝叶斯滤波的推导 粒子滤波&#xff08;Particle filter&#xff09;特性流程状态传播权重评估重采样算法流程存在的问题 FastSLAM的原理及优化FastSLAM介绍算法流程FastSLAM优化存在的问题及优化进一步优化proposal分布最终…

RabbitMQ学习整理————基于RabbitMQ实现RPC

基于RabbitMQ实现RPC 前言什么是RPCRabbitMQ如何实现RPCRPC简单示例通过Spring AMQP实现RPC 前言 这边参考了RabbitMQ的官网&#xff0c;想整理一篇关于RabbitMQ实现RPC调用的博客&#xff0c;打算把两种实现RPC调用的都整理一下&#xff0c;一个是使用官方提供的一个Java cli…

思维模型整合

思维模型整合 4P--- 4C思考模型能力圈模型 4P— 4C思考模型 在竞争激烈的今天&#xff0c;每个赛道都有众多可以为客户提供相同价值的对手&#xff0c;而赛道中的佼佼者之所以能打败大部分人&#xff0c;可能并不是他们能比别人更能讨好大众&#xff0c;而是因为在这个赛道它有…

Jmeter学习系列之六:阶梯加压线程组Stepping Thread Group详解

性能测试中,有时需要模拟一种实际生产中经常出现的情况,即:从某个值开始不断增加压力,直至达到某个值,然后持续运行一段时间。 在jmeter中,有这样一个插件,可以帮我们实现这个功能,这个插件就是:Stepping Thread Group 1、下载配置方法 1.1.下载配置 插件下载地址:…

Qt读写局域网共享文件夹

Qt读写局域网共享文件夹 #include "widget.h" #include "ui_widget.h"#include <QDir> #include <QSettings>//Qt读取局域网共享文件夹文件内容 解决不同网段无法访问共享文件夹问题 //使用QDir或QFile等都可以直接访问局域网共享文件. //路径…

Selenium定位不到元素怎么办?一定要这么做

在使用Selenium进行自动化测试时&#xff0c;碰到无法定位元素该怎么办&#xff1f;这里总结了9种情况下的元素定位方法&#xff1a; 1、frame/iframe表单嵌套 WebDriver只能在一个页面上对元素识别与定位&#xff0c;对于frame/iframe表单内嵌的页面元素无法直接定位。 解决…

Jenkins持续集成Python项目

一、前言   之前学习了很多自动化测试框架&#xff0c;但是写的脚本都是本地执行&#xff0c;多数用来造数据。最近公司掀起一股自动化测试的风&#xff0c;所以就想研究下如何集成jenkins&#xff0c;本次采用pytest&#xff0c;用的是阿里云服务器centos7。 二、服务器环境…

石头剪刀布游戏(C语言)

题目描述 石头剪刀布游戏有 3 种出拳形状&#xff1a;石头、剪刀、布。分别用字母 A , B , C 表示。 游戏规则: 出拳形状之间的胜负规则如下&#xff1a; A > B&#xff1b;B > C&#xff1b;C > A&#xff1b;">"左边一个字母&#xff0c;表示相对优…

bat 查找文件所在

脚本 在批处理文件&#xff08;.bat&#xff09;中查找文件所在的目录&#xff0c;你可以使用dir命令结合循环和条件语句来实现。以下是一个简单的示例&#xff0c;演示如何在批处理文件中查找指定文件并输出其所在目录&#xff1a; echo off setlocal enabledelayedexpansio…

Vue 封装的 axios 类的使用(小bug 改进)

http类 import { baseUrl } from "./config"; //引入config.js中的配置 import axios from "axios"; //引入axios import qs from "querystringify"; //form-Data请求时的工具类class Http{axios null;lastRequestIntercept null…

Linux安全基线与加固

基于CIS 基线 GitHub - daniel-armbrust/linux-security-baseline: Linux Security Baseline based on CIS Benchmarks. |----------------[ GNU/Linux安全基线与加固-0.3 ]----------------|0. About this doc1. Routine security baseline1.1 Security fix update1.2 Pass…

开源LLMs导览:工作原理、顶级LLM列表对比

目录 一、开源 LLM 是什么意思&#xff1f;二、开源LLM如何工作&#xff1f;2.1 预训练2.2 代币化2.3 开源LLM的微调2.4 输入编码2.5 训练与优化2.6 推理 三、开源LLM对组织的好处3.1 增强的数据安全和隐私3.2 节约成本3.3 减少供应商依赖性3.4 代码透明度 四、哪种LLM模式最好…

数据可视化在商业领域有哪些重要性?

数据可视化在商业领域的重要性体现在多个方面&#xff0c;它通过将复杂的数据集转化为直观、易于理解的图形和图表&#xff0c;帮助企业和组织做出更明智的决策。以下是数据可视化对商业的一些关键重要性&#xff1a; 提高决策效率&#xff1a;通过直观的图表和图形&#xff0c…

漫漫数学之旅031

文章目录 经典格言数学习题古今评注名人小传 - 经典格言 如果没有数学知识&#xff0c;这个世界的事物是无法搞清楚的。——罗杰培根&#xff08;Roger Bacon&#xff09; 好的&#xff0c;各位看官&#xff0c;让我们来听听罗杰培根这位中世纪的“科学老顽童”是怎么说的&…

罗克韦尔AB的PLC实现ModbusTCP和ModbusRTU协议标签方式通讯

本文是通过IGT-DSER智能网关读写AB罗克韦尔Compact、Control系列PLC的标签数据缓存并转为Modbus从站协议&#xff0c;与上位机通讯的案例。 打开智能网关的参数软件(下载地址)&#xff0c;通过功能->数据转发与平台对接&#xff0c;再选择数据转发与缓存’&#xff0c;进入以…