数据结构——栈的实现

server/2025/1/11 11:52:50/

今天,我们来写一下关于栈的博文。

1.首先我们先了解一下什么是栈?

一:概念:

:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。

进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。

栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。

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

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

为了更好的理解,我们画个图辅助了解一下吧。

具体解释:

你看上面:

想要加数据的时候,就在栈顶入数据,此时叫压栈

再看,想要删数据的时候,是不是也得再栈顶出?此时叫出栈

同时,你是不是就发现了无论它是加还是删,遵循的规律都是后进先出

 

二:栈的实现

好了,有了上面的认识后,就可以进行实现部分了。

在此之前,我们首先得考虑考虑,我们得采用什么形式呢?是数组还是链表呢?

一般来说,两者都是可以的,但是使用数组的形式更优一些,对此我们在这里就以链表来实现它。
链式栈:可以用双向链表,但最好用单链表(用头那边当栈顶)
如果用的是数组的话,一般是尾部当栈顶

其实,感觉这里跟顺序表那些都差不多呢~

好了,至此,正式开始吧:

建立数组栈结构:

typedef struct Stack
{STDataType* a;int top;int capacity;
}ST;

初始化部分

void STInit(ST* ps)
{assert(ps);ps->a = (STDataType*)malloc(sizeof(STDataType) * 4);if (ps->a == NULL){perror("malloc fail");return;}ps->capacity = 4;ps->top = 0;   // top是栈顶元素的下一个位置//ps->top = -1;   // top是栈顶元素位置
}


但是这里有些问题:top指向的是栈顶的下一个,先放数据再++
还是栈顶位置,一开始就在0这个位置出发。先++再放数据。给-1位置

看下图:

那么,肯定有人疑惑为什么呢?

答:因为当初始化给0时,就代表栈顶有元素了,给-1的时候就表示栈为空。

你也可以那样想到数组那个,你看:

数组[0]的时候它是不是就代表已经有一个元素了,这个元素相当于就在第一个位置那样?

同样,当你希望初始化时不需要有数据(相当于联想到数组),当0时有一个数字,这时,你给-1,是不是相当于没有数据了?

有人有问了,当你给-1的话,不会变成野指针了吗?

其实这个问题不用担心,因为这里给-1的时候就要判断了。这里0去访问是合法的,但是刚初始化没有push的内容,就会有问题了,所以如果使用0的话,还需要做出处理。

总的来说,使用0还是-1都是没有强制性要求你,都是可以的,看个人喜好了。

这里使用的0的形式。

销毁部分:

void STDestroy(ST* ps)
{assert(ps);free(ps->a);ps->a = NULL;ps->top = 0;ps->capacity = 0;
}

这里销毁的时候,使用-1和0,ps->top就会有一点的区别了。

插入部分

void STPush(ST* ps, STDataType x)
{assert(ps);if (ps->top == ps->capacity){STDataType* tmp = (STDataType*)realloc(ps->a,sizeof(STDataType) * ps->capacity*2);if (tmp == NULL){perror("realloc fail");return;}ps->a = tmp;ps->capacity *= 2;}ps->a[ps->top] = x;ps->top++;
}

解释:

在插入之前,我们先进行判断它是否已经满了,满了的话就扩容,反之不用。

插入的话,也比较简单,由于它只能在栈顶插入嘛。因为这里使用的0的形式

所以就直接找到栈顶的位置,插进去就可以了。

ps->a[ps->top] = x;ps->top++;

删除部分

void STPop(ST* ps)
{assert(ps);assert(!STEmpty(ps));ps->top--;
}

也由于只能栈顶先删,所以就减减top就可以了。

得出栈的大小部分

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

因为使用的是数组栈,我们又知道,数组是连续的,所以直接返回top所在的位置,就可以得出大小了。

判断空部分

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

因为使用的是0,即栈的下一个,当它栈顶的位置是0的时候,就代表它没有数据了,也就是空。

此时返回就好了。

另外,我们上面的删是不是也得判断一下是否空的情况对不对?如果它已经空了,那么还删的话,有什么意义?对吧?

所以可以看到上面的删除部分,我们使用了断言它不为空。

找尾部分

STDataType STTop(ST* ps)
{assert(ps);assert(!STEmpty(ps));return ps->a[ps->top - 1];
}

这是说,如果你要找尾的话,数组嘛,所以直接找下标就可以了。

好了,功能已经介绍完了。

现在来附上完整代码:

Stack部分

#define _CRT_SECURE_NO_WARNINGS 1
#include "Stack.h"void STInit(ST* ps)
{assert(ps);ps->a = (StDatatype*)malloc(sizeof(StDatatype) * 4);if (ps->a == NULL){perror("malloc fail");return;}ps->top = 0;ps->capacity = 4;
}void Destory(ST* ps)
{assert(ps);free(ps->a);ps->a = NULL;ps->capacity = 0;ps->top = 0;
}void STPush(ST* ps,StDatatype x)
{if (ps->top== ps->capacity){StDatatype* temp =(StDatatype*) realloc(ps->a, sizeof(StDatatype) * ps->capacity * 2);if (temp == NULL){perror("realloc fail");return 0;}ps->a = temp;ps->capacity *= 2;}ps->a[ps->top] = x;ps->top++;
}
void STPop(ST* ps)
{assert(ps);assert(!STEmpty(ps));ps->top--;
}
int size(ST* ps)
{assert(ps);return ps->top;
}
bool STEmpty(ST* ps)
{assert(ps);return ps->top == 0;
}
StDatatype STTop(ST* ps)
{assert(ps);assert(!STEmpty(ps));return ps->a[ps->top-1];
}

Stack.h部分

#pragma once
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<stdbool.h>
//
//typedef struct Stack
//{
//	int a[20];
//	int top;
//};typedef char StDatatype;
typedef struct Stack
{StDatatype* a ;int top;int capacity;}ST;//ʼ
void STInit(ST* ps);
void Destory(ST* ps);
void STPush(ST* ps,StDatatype x);
void STPop(ST* ps);
int size(ST* ps);
bool STEmpty(ST* ps);
StDatatype STTop(ST* ps);
bool isValid(char* s);

测试部分:

#define _CRT_SECURE_NO_WARNINGS 1
#include "Stack.h"
int main()
{ST st;STInit(&st);STPush(&st,'(');STPush(&st, ']');bool ret = isValid(&s);if (ret == "false"){printf("no");}else{printf("yes");}Destory(&st);return 0;
}

好了,到了每次鸡汤部分:

你每一步都会算数的!


http://www.ppmy.cn/server/157456.html

相关文章

英雄联盟丢失dll文件怎么解决?游戏中发现dll找不到的处理方法

下班或放学回到家&#xff0c;美滋滋地打开电脑&#xff0c;准备在召唤师峡谷大杀四方&#xff0c;结果点击启动游戏的瞬间&#xff0c;一个弹窗如同 “恶魔” 般出现 ——“XX.dll 文件丢失&#xff0c;无法启动游戏”。那一刻&#xff0c;是不是感觉心都凉了半截&#xff1f;…

英语互助小程序springboot+论文源码调试讲解

第2章 开发环境与技术 英语互助小程序的编码实现需要搭建一定的环境和使用相应的技术&#xff0c;接下来的内容就是对英语互助小程序用到的技术和工具进行介绍。 2.1 MYSQL数据库 本课题所开发的应用程序在数据操作方面是不可预知的&#xff0c;是经常变动的&#xff0c;没有…

风水算命系统架构与功能分析

系统架构 服务端&#xff1a;Java&#xff08;最低JDK1.8&#xff0c;支持JDK11以及JDK17&#xff09;数据库&#xff1a;MySQL数据库&#xff08;标配5.7版本&#xff0c;支持MySQL8&#xff09;ORM框架&#xff1a;Mybatis&#xff08;集成通用tk-mapper&#xff0c;支持myb…

企业通过私有安全端点访问大型语言模型的益处

企业通过私有安全端点访问大型语言模型的益处 随着大型语言模型&#xff08;LLMs&#xff09;如 GPT-4、LLaMA、BARD、Falcon 和 Claude 等技术的迅速发展&#xff0c;企业在利用人工智能&#xff08;AI&#xff09;优化其业务流程、生成类人文本、回答问题和总结文档等方面的…

Go语言中的sync.WaitGroup详解

Go 语言作为一种现代并发编程语言,提供了强大的并发模型和工具。其中,sync.WaitGroup 是 Go 标准库中的一个重要同步工具,广泛用于协程(goroutine)的同步控制。本文将深入探讨 sync.WaitGroup 的工作原理、应用场景以及如何避免使用共享变量和信号量来实现同步。 一、syn…

Springboot Rabbitmq + 线程池技术控制指定数量task执行

定义DataSyncTaskManager&#xff0c;作为线程池任务控制器 package org.demo.scheduletest.service;import lombok.extern.slf4j.Slf4j;import java.util.concurrent.BlockingQueue; import java.util.concurrent.Executors; import java.util.concurrent.LinkedBlockingQueu…

【Linux】Linux开发:GDB调试器与Git版本控制工具指南

Linux相关知识点可以通过点击以下链接进行学习一起加油&#xff01;初识指令指令进阶权限管理yum包管理与vim编辑器GCC/G编译器make与Makefile自动化构建 在 Linux 开发中&#xff0c;GDB 调试器和 Git 版本控制工具是开发者必备的利器。GDB 帮助快速定位代码问题&#xff0c;G…

【学习笔记】数据结构(十二)

文件 文章目录 文件12.1 有关文件的基本概念12.2 顺序文件12.3 索引文件12.4 ISAM文件和VSAM文件12.4.1 ISAM文件12.4.2 VSAM文件 12.5 直接存取文件(散列文件)12.6 多关键字文件12.6.1 多重表文件12.6.2 倒排文件 12.1 有关文件的基本概念 文件(file) 是由大量性质相同的记录…