栈和队列(8.4)

news2024/12/28 3:32:33

1.
1.1 栈的概念及结构
栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。 进行数据插入和删除操作的一端
称为栈顶,另一端称为栈底。 栈中的数据元素遵守后进先出 LIFO Last In First Out )的原则。
压栈:栈的插入操作叫做进栈 / 压栈 / 入栈, 入数据在栈顶
出栈:栈的删除操作叫做出栈。 出数据也在栈顶

后入栈的数据先出来,例如存入1,2,3,出栈的顺序是3,2,1。

1.2 栈的实现

思考一下,栈的实现是用数组还是链表更合适呢?

都可以实现,但相对来说,数组更好一些,数组的尾插尾删效率更高。

typedef int STDataType;
//支持动态增长的栈
typedef struct Stack
{
    STDataType* a;
    int top;//栈顶
    int capacity;//容量
}ST;

1.2.1 栈初始化

存在一个问题,栈顶 top 应该初始化为多少?

如果初始化为 0,入栈一个数据 top ++,意味着 top 是栈顶元素的下一个位置

初始化为 -1, top 则为栈顶元素本身

这两种做法都是可以的。

完整代码:
头文件:

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
//定义栈

静态栈(一般不用)
//#define N 10
//struct Stack
//{
//	int a[N];
//	int top;
//};

typedef int STDataType;
//支持动态增长的栈
typedef struct Stack
{
	STDataType* a;
	int top;//栈顶
	int capacity;//容量
}ST;

// 初始化栈
void STInit(ST* ps);
// 入栈
void STPush(ST* ps, STDataType x);
// 出栈
void STPop(ST* ps);
// 获取栈顶元素
STDataType STTop(ST* ps);
// 获取栈中有效元素个数
int STSize(ST* ps);
// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
bool STEmpty(ST* ps);
// 销毁栈
void STDestroy(ST* ps);

测试文件:

#define _CRT_SECURE_NO_WARNINGS 1
#include"Stack.h"

void TestStack1()
{
	//初始化结构体
	ST st;
	STInit(&st);
	STPush(&st, 1);
	STPush(&st, 2);
	STPush(&st, 3);
	STPush(&st, 4);
	//链表不为空
	while (!STEmpty(&st))
	{
		//打印栈顶
		printf("%d ",STTop(&st));
		//打印一个,出栈一个
		STPop(&st);
	}
        printf("\n");

		STDestroy(&st);
}

int main()
{
	TestStack1();

	return 0;
}

实现文件:

#define _CRT_SECURE_NO_WARNINGS 1
#include"Stack.h"

// 初始化栈
void STInit(ST* ps)
{
	assert(ps);
	ps->a = NULL;
	ps->capacity = 0;
	ps->top = 0;
}

// 销毁栈
void STDestroy(ST* ps)
{
	assert(ps);
	//释放
	free(ps->a);
	ps->a = NULL;
	ps->top = ps->capacity = 0;
}

// 入栈
void STPush(ST* ps, STDataType x)
{
	assert(ps);
	//如果放满,扩容
	if (ps->top == ps->capacity)
	{
		//如果容量为空,就先赋值,如果不为空,就将容量翻倍
		int newCapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
		//直接用realloc申请空间,当原本没有空间时,它等同于malloc
		//先用tmp承接开辟的空间,以免开辟失败破坏原空间
		STDataType* tmp = (STDataType*)realloc(ps->a,sizeof(STDataType) * newCapacity);
		if (tmp == NULL)
		{
			perror("realloc");
			exit(-1);
		}
		ps->a = tmp;
		ps->capacity = newCapacity;
        
	}
        //放值
		ps->a[ps->top] = x;
		ps->top++;
}

// 出栈
void STPop(ST* ps)
{
	assert(ps);
	//top为0说明栈空,不能继续删
	assert(ps->top > 0);

	--ps->top;
}

// 获取栈中有效元素个数
int STSize(ST* ps)
{
	assert(ps);
	//top是栈顶元素的下一个位置,正好是size
	return ps->top;
}

// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
bool STEmpty(ST* ps)
{
	assert(ps);
	//top为0栈为空
	return ps->top == 0;
}

// 获取栈顶元素
STDataType STTop(ST* ps)
{
	assert(ps);
	//top为0说明栈空,没有元素
	assert(ps->top > 0);
	return ps->a[ps->top - 1];
}

例题:

20. 有效的括号 - 力扣(LeetCode)

思路:由于左括号需要以正确的顺序闭合,栈的特性完美适配这一要求:我们创建一个栈(栈的基本操作我们直接cv前面的代码),将左括号入栈,然后取栈顶,与栈外的右括号相匹配,如果刚好完全匹配没有剩余,则为有效,相反,左括号或右括号中任一有剩余则为无效。


typedef char STDataType;
//支持动态增长的栈
typedef struct Stack
{
	STDataType* a;
	int top;//栈顶
	int capacity;//容量
}ST;

// 初始化栈
void STInit(ST* ps);
// 入栈
void STPush(ST* ps, STDataType x);
// 出栈
void STPop(ST* ps);
// 获取栈顶元素
STDataType STTop(ST* ps);
// 获取栈中有效元素个数
int STSize(ST* ps);
// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
bool STEmpty(ST* ps);
// 销毁栈
void STDestroy(ST* ps);
// 初始化栈
void STInit(ST* ps)
{
	assert(ps);
	ps->a = NULL;
	ps->capacity = 0;
	ps->top = 0;
}

// 销毁栈
void STDestroy(ST* ps)
{
	assert(ps);
	//释放
	free(ps->a);
	ps->a = NULL;
	ps->top = ps->capacity = 0;
}

// 入栈
void STPush(ST* ps, STDataType x)
{
	assert(ps);
	//如果放满,扩容
	if (ps->top == ps->capacity)
	{
		//如果容量为空,就先赋值,如果不为空,就将容量翻倍
		int newCapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
		//直接用realloc申请空间,当原本没有空间时,它等同于malloc
		//先用tmp承接开辟的空间,以免开辟失败破坏原空间
		STDataType* tmp = (STDataType*)realloc(ps->a,sizeof(STDataType) * newCapacity);
		if (tmp == NULL)
		{
			perror("realloc");
			exit(-1);
		}
		ps->a = tmp;
		ps->capacity = newCapacity;
        
	}
        //放值
		ps->a[ps->top] = x;
		ps->top++;
}

// 出栈
void STPop(ST* ps)
{
	assert(ps);
	//top为0说明栈空,不能继续删
	assert(ps->top > 0);

	--ps->top;
}

// 获取栈中有效元素个数
int STSize(ST* ps)
{
	assert(ps);
	//top是栈顶元素的下一个位置,正好是size
	return ps->top;
}

// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
bool STEmpty(ST* ps)
{
	assert(ps);
	//top为0栈为空
	return ps->top == 0;
}

// 获取栈顶元素
STDataType STTop(ST* ps)
{
	assert(ps);
	//top为0说明栈空,没有元素
	assert(ps->top > 0);
	return ps->a[ps->top - 1];
}
bool isValid(char * s){
     ST st;
     char top;
     STInit(&st);
     while(*s)
     {
         //如果为左括号
         if(*s=='{'||*s=='('||*s=='[')
         {
             //入栈
             STPush(&st, *s);
         }
         //如果为右括号
         else
         {
             //栈内为空,说明左右括号数量不匹配
             if(STEmpty(&st))
             {
                 //销毁空间,返回
                 STDestroy(&st);
                 return false;
             }
             //栈内不为空,取栈顶与栈外的右边括号匹配
             top=STTop(&st);
             //取出之后就出栈,以便于后面的元素出栈
             STPop(&st);
             //左右不匹配
             if(*s=='}'&&top!='{'
             ||*s==']'&&top!='['
             ||*s==')'&&top!='(')
             {
                 STDestroy(&st);
                 return false;
             }
         }
         s++;
     }
     //通过遍历说明右括号完全被匹配,栈内的左括号可能有剩余,没有剩余ture,有剩余false
     bool ret=STEmpty(&st);
     return ret;
     STDestroy(&st);
}

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

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

相关文章

移远通信联合产业中坚力量共同发起倡议,推动5G RedCap技术演进和应用创新发展

10月20-21日&#xff0c;2023年中国5G发展大会在上海召开。会议第二日&#xff0c;“5G轻量化&#xff08;RedCap&#xff09;技术演进和应用创新发展”论坛同步举办。 作为5G及RedCap技术发展的先行力量&#xff0c;移远通信受邀出席论坛。同时&#xff0c;公司副总经理刘明辉…

Go 类型全解:常量与变量大全!

一、类型确定值 类型确定值是与特定数据类型明确绑定的。类型确定值在 Go 中占有很大一部分领域&#xff0c;包括但不限于常量、变量、函数返回值、结构体字段等。下面是对类型确定值的的示例&#xff1a; 类型确定值在变量声明中 当你在变量声明中明确指定了类型&#xff0…

PLC MODBUS-ASCII协议通信( LRC校验码SCL计算FC)

Modbus-RTU协议在RS485总线上有RTU和ASCII两种传输格式,其中RTU协议应用比较多,ASCII协议很少使用。有些仪表可能会使用到ASCII协议,这篇博客我们简单介绍下MODBUS-ASCII协议,ASCII协议使用的是文本传输,整个数据包是可打印字符。 帧头一般是冒号: 帧尾是换行符\r\n。 有…

【每日一题】老人的数目

文章目录 Tag题目来源题目解读解题思路方法一&#xff1a;遍历 其他语言python3 写在最后 Tag 【遍历】【数组】【2023-10-23】 题目来源 2678. 老人的数目 题目解读 找出乘客中年龄严格大于 60 的人数。 解题思路 方法一&#xff1a;遍历 本题比较简单&#xff0c;直接遍…

智慧燃气巡检管理系统

我们知道燃气设施的巡检、巡查是运维工作中一项重要的基础工作&#xff0c;而巡检人员主要靠手动记录&#xff0c;回到公司后还得再进行录入归档、导入照片&#xff0c;然后打印装订等&#xff0c;涉及工作量也是不小的&#xff1b;还有人员更替&#xff0c;易造成人员对燃气设…

【设计模式】概括

设计模式 什么是设计模式 设计模式&#xff0c;是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性、程序的重用性。 设计模式分类 创建型模式&#xff0c;共五种&#xff1a;工厂…

Java方法区

方法区 Java方法区&#xff08;Method Area&#xff09;&#xff0c;在Java虚拟机&#xff08;JVM&#xff09;内存结构中是一个非常重要的组成部分。方法区是用来存储类信息、常量、静态变量以及即时编译器编译后的代码等数据的内存区域。 内部结构 类元数据&#xff08;Cl…

Python数据挖掘 | 升级版自动查核酸

&#x1f4d5;作者简介&#xff1a;热爱跑步的恒川&#xff0c;致力于C/C、Java、Python等多编程语言&#xff0c;热爱跑步&#xff0c;喜爱音乐的一位博主。 &#x1f4d7;本文收录于恒川的日常汇报系列&#xff0c;大家有兴趣的可以看一看 &#x1f4d8;相关专栏C语言初阶、C…

常见的状态转移矩阵和对应的运动模型

状态转移矩阵的形式取决于我们所建模的系统的动态特性。对于不同的运动模型&#xff0c;状态转移矩阵将会有所不同。以下是一些常见的状态转移矩阵和对应的运动模型&#xff1a; 恒定速度模型&#xff1a; 这是你给出的模型&#xff0c;其中物体假设以恒定速度移动。 恒定加速…

图书推荐管理系统Python+Django网页界面+协同过滤推荐算法

一、介绍 图书管理与推荐系统。使用Python作为主要开发语言。前端采用HTML、CSS、BootStrap等技术搭建界面结构&#xff0c;后端采用Django作为逻辑处理&#xff0c;通过Ajax等技术实现数据交互通信。在图书推荐方面使用经典的协同过滤算法作为推荐算法模块。主要功能有&#…

【python海洋专题二十八】南海四季海流流速图

【python海洋专题二十八】南海四季海流流速图 往期推荐 图片 【python海洋专题一】查看数据nc文件的属性并输出属性到txt文件 【python海洋专题二】读取水深nc文件并水深地形图 【python海洋专题三】图像修饰之画布和坐标轴 【Python海洋专题四】之水深地图图像修饰…

云HIS系统,Cloud HIS system,云HIS医院信息管理系统源码

通过云HIS平台,可以减少医院投资,无需自建机房和系统,快速实现信息化服务。系统升级及日常维护服务有云平台提供,无需配备专业IT维护人员进行系统维护。 一、his系统和云his系统的区别 His系统和云his系统是两种不同的计算平台&#xff0c;它们在技术架构上存在很大的差异。下…

【python海洋专题二十七】南海四季海流图

【python海洋专题二十七】南海四季海流图 往期推荐 **[[ 【python海洋专题一】查看数据nc文件的属性并输出属性到txt文件] 【python海洋专题二】读取水深nc文件并水深地形图 【python海洋专题三】图像修饰之画布和坐标轴 【Python海洋专题四】之水深地图图像修饰 【Pyth…

电动汽车交流充电桩系统的设计方案

摘要&#xff1a;作为新能源汽车的基础动力装置&#xff0c;交流充电桩也是可以促使新能源汽车正常行驶的关键内容。与其他汽车不同的是&#xff0c;新能源汽车并不需要汽油维持其运行&#xff0c;只需要充电就可以保证汽车行驶的需求&#xff0c;可以降低汽油排放对环境的污染…

SpringBoot项目把Mysql从5.7升级到8.0

首先你需要把之前的库导入到mysql库导入到8.0的新库中。&#xff08;导入的时候会报错我是通过navcat备份恢复的&#xff09; 1、项目中需要修改pom文件的依赖 mysql 和 jdbc <dependency><groupId>mysql</groupId><artifactId>mysql-connector-java&…

nginx配置负载均衡--实战项目(适用于轮询、加权轮询、ip_hash)

&#x1f468;‍&#x1f393;博主简介 &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01; &#x1f40b; 希望大家多多支…

html登录注册标签

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> <body> <h1>登录注册</h1> <form action"第一个网页.html" method"post&quo…

LeetCode算法心得——元素和最小的山形三元组 II(预处理和简单动规)

大家好&#xff0c;我是晴天学长&#xff0c;枚举&#xff0b;简单的动态规划思想&#xff0c;和前段时间的周赛题的写法可以说一模一样&#xff0c;像这种类似3元的题&#xff0c;要控制时间复杂度的话&#xff0c;只能枚举一个变量&#xff0c;所以要前缀和或者动规等待。需要…

【Java笔试强训】Day3(OR59 字符串中找出连续最长的数字串、JZ39 数组中出现次数超过一半的数字)

OR59 字符串中找出连续最长的数字串 链接&#xff1a;OR59 字符串中找出连续最长的数字串 题目&#xff1a; 读入一个字符串str&#xff0c;输出字符串str中的连续最长的数字串 题目分析&#xff1a; 代码实现&#xff1a; package Day3;import java.util.Scanner;public…

k8s 使用ingress-nginx访问集群内部应用

k8s搭建和部署应用完成后&#xff0c;可以通过NodePort&#xff0c;Loadbalancer&#xff0c;Ingress方式将应用端口暴露到集群外部&#xff0c;提供外部访问。 缺点&#xff1a; NodePort占用端口&#xff0c;大量暴露端口非常不安全&#xff0c;并且有端口数量限制【不推荐】…