【数据结构和算法】--- 栈

news2024/10/5 1:03:24

目录

  • 栈的概念及结构
  • 栈的实现
    • 初始化栈
    • 入栈
    • 出栈
    • 其他一些栈函数
  • 小结
  • 栈相关的题目

栈的概念及结构

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

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

联想一下,其实栈就相当于手枪的弹夹,将子弹压入弹夹的操作就相当于压栈,打出子弹的操作就相当于出栈,每次先打出的子弹都是我们最后压入弹夹的子弹,最后一颗子弹就是我们最先压入的那一颗了,这就是后进先出。栈也如此,结构大致如下:

在这里插入图片描述

基于这样的结构,那么如果我们想要拿到栈的某个元素,就必须要先把此元素以上的元素依次出栈,然后才能取出。

栈的实现

两种方式可以实现栈结构-数组栈,链式栈

  1. 链式栈
    如果用单链表实现:若栈底就指向头节点,栈顶就指向尾节点。这样设计入栈很方便,相当于头插,时间复杂度为O(1);但出栈操作就必须要先遍历链表找到栈顶的前一个,然后再出栈,并修改栈顶,相当于尾删,时间复杂度达到O(N)于是乎我们一般将栈顶指向头节点,栈底指向尾节点,这样入栈出栈就都是O(1)了,即头插/头删。

在这里插入图片描述

如果用双向链表实现:栈顶为链表的头和尾都可以,入栈和出栈时间复杂度都为O(1),但双向链表结构较为复杂,一般不选用此结构

  1. 数组栈
    数组栈的入栈和出栈的实现较为简单,且时间复杂度为O(1)

在这里插入图片描述

相较于链式栈,数组栈访问数据时cpu缓存命中率比较高但链式栈相较于数组栈也会节省一定的空间。下面栈的实现主要用的是数组栈。
通常我们标识栈顶位置的下一个位置为top(即下标为size的位置)。与标识栈顶位置为top相比较,这样可以减少栈为空,栈容量判断等函数的难度,且若标识栈顶位置为top,当栈里面没有元素时top的指向也较为尴尬。
我们可以如下定义栈结构:

typedef int STDataType;
//数组栈
typedef struct stack
{
	STDataType* a;
	int top;//标识栈顶下一个元素下标(同为栈元素个数)
	int capacity;
}ST;

初始化栈

通过上面对栈的介绍进行初始化。

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

入栈

入栈操作就是向数组内增加一个数,首先要判断栈(数组)容量pst->capacity是否需要增容,然后top位置(即pst->a[top])增加一个数,最后重新变换top指向(即pst->top++),具体如下:

//入栈
void StackPush(ST* pst, STDataType x)
{
	assert(pst);
	//判断增容
	if (pst->top == pst->capacity)
	{
		int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;
		STDataType* newnode = (STDataType*)realloc(pst->a, sizeof(ST) * newcapacity);
		if (newnode == NULL)
		{
			perror("check_ST_capacity()::malloc");
			return;
		}
		pst->a = newnode;
		pst->capacity = newcapacity;
	}
	//目标数x入栈
	pst->a[pst->top] = x;
	//变换top指向
	pst->top++;
}

出栈

出栈操作就相对简单了,直接改变top指向就可以了(即pst->top--)。如果栈里面已经没有元素了,那执行此操作top指向就会错误,于是乎我们需要断言一下来确保栈里面有元素可以删除(即assert(ps->top != 0);)。

//出栈
void StackPop(ST* pst)
{
	assert(pst);
	assert(pst->top != 0);
	pst->top--;
}

其他一些栈函数

  1. 获得栈顶元素:
    pst->top指向的是栈顶的下一个元素的下标,那么只需要让他--即可(即pst->a[pst->top-1]),在使用前确保栈中有元素,不然程序会崩溃(越界访问)
// 获取栈顶元素 
STDataType StackTop(ST* pst)
{
	assert(pst);
	assert(pst->top != 0);
	return pst->a[pst->top - 1];
}
  1. 获得栈有效元素个数:
    pst->top指向的既是指向栈顶下一个元素的下标也是整个栈里面有效数据的个数,所以此函数返回pet->top即可。
// 获取栈中有效元素个数
int StackSize(ST* pst)
{
	assert(pst);
	return pst->top;
}
  1. 检查栈是否为空:
    同理只要栈里面有效元素个数为0,那么栈就是空栈,如下:
// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0 
bool StackEmpty(ST* pst)
{
	assert(pst);
	return pst->top == 0;
}
  1. 栈的销毁:
    栈的销毁本质上是释放先前realloc()开辟的数组,再将容量和栈顶置0即可。
// 销毁栈 
void StackDestroy(ST* pst)
{
	assert(pst);
	assert(pst->capacity != 0);
	free(pst->a);
	pst->a = NULL;
	pst->top = pst->capacity = 0;
}

小结

  • 栈是一种后进先出的结构,这一点恰与我们后面要讲的队列相反;

  • 顺序表和链表都可以用来实现栈,不过一般都使用顺序表,因为栈想当于是阉割版的顺序表,只用到了顺序表的尾插和尾删操作,顺序表的尾插和尾删不需要搬移元素,因此效率非常高O(1),故一般都是使用顺序表实现;

  • 栈结构中的top一般为要插入位置的下标(即栈顶元素下一个位置),这是为了方便区分栈为空栈的情况,且后续函数更好实现;

  • 栈只能在栈顶进行输入的插入和删除操作,不支持随机访问;


栈相关的题目

  1. 关于入栈和出栈顺序,如下:

若进栈序列为 1,2,3,4 ,进栈过程中可以出栈,则下列不可能的一个出栈序列是()
A 1,4,3,2
B 2,3,4,1
C 3,1,4,2
D 3,4,2,1

不难看出是c选项错了,因为如果第一个出栈的是3,那么在3之前压栈的12就都还没有出栈,所以接下来出栈的只能有两种情况:

  • 1.4接着入栈然后出栈,即为D选项;
  • 2.直接出先前压栈的2

对于C选项,此时的1还在栈底,在它上面还有2,所以不能直接出1

  1. LeetCode OJ题: 有效的括号

题目描述:给定一个只包括 '('')''{''}''['']'的字符串s ,判断字符串是否有效。
有效字符串需满足:

  • 左括号必须用相同类型的右括号闭合。
  • 左括号必须以正确的顺序闭合。
  • 每个右括号都有一个对应的相同类型的左括号。

这题主要考察我们对栈特性的应用,即后进先出,那么我们便可这样设计:循环遍历字符串s中的每个字符,满足以下条件的对栈进行入/出栈操作:

  1. 遇到左括号,直接入栈
  2. 遇到右括号,取栈顶元素进行匹配,若不匹配直接返回false,若匹配就将此括号出栈,并继续循环。

另外我们还要对如下两种情况做出判断

  1. 当遍历到右括号时,此时栈中是否还有元素?(QueueEmpty()?)为空直接返回false
  2. 当字符串s遍历结束时,栈中是否还有剩余元素?(QueueEmpty()?)不为空直接返回false,为空返回true

其中一些栈的接口函数就不展示了,上面内容都有,代码实现如下:

bool isValid(char* s)
{
    ST st;//创建栈
    StackInit(&st);//初始化栈
    //遍历字符串s
    while(*s)
    {
        if(*s == '(' || *s == '[' || *s == '{')
        {
            StackPush(&st,*s);
        }
        else
        {
            //栈为空判断,为空返回false,如上讲解1处
            if(StackEmpty(&st))
            {
                StackDestroy(&st);
                return false;
            }
            char ch = StackTop(&st);
            //左右括号匹配判断,匹配错误返回false
            if((*s == ')' && ch != '(') || 
               (*s == ']' && ch != '[') ||
               (*s == '}' && ch != '{'))
                {
                    StackDestroy(&st);
                    return false;
                }
            StackPop(&st);
        }
        s++;
    }
    //栈为空判断,不为空返回false,与上面判断处区分,如上讲解2处
    if(!StackEmpty(&st))
    {
        StackDestroy(&st);
        return false;
    }
    return true;
}

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

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

相关文章

机器人学习目标

学习目标: 若干年后,我们都将化为尘土,无人铭记我们的存在。那么,何不趁现在,尽己所能,在这个世界上留下一些痕迹,让未来的时光里,仍有人能感知到我们的存在。 机器人协会每届每个阶…

【开源】基于Vue.js的公司货物订单管理系统

文末获取源码,项目编号: S 082 。 \color{red}{文末获取源码,项目编号:S082。} 文末获取源码,项目编号:S082。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 客户管理模块2.2 商品维护模块2.3 供…

Ultimate VFX

Ultimate VFX 构建套件:

C++实现2024新年日历

日历计算 平年润年计算 首先大概日历最麻烦的就是2月份的计算了,因为会需要考虑到平年还是闰年。每年的2月份天数的计算方法,如果年份能被4整除但不能被100整除,或者能被400整除,那么这一年的2月份只有29天(闰年&…

idea import 不让打包时自动带*

场景 因为多人开发,合代码时经常发现因自动import带*了,导致各种问题。 解决

docker学习(八、mysql8.2主从复制遇到的问题)

在我配置主从复制的时候,遇到了一直connecting的问题。 起初可能是我ip配置的不对,slave_io_running一直connecting。(我的环境:windows中安装了wsl,是ubuntu环境的,在wsl中装了miniconda,mini…

为什么要使用AI伪原创工具?盘点最新的几款AI伪原创工具

在数字时代,内容创作已成为广大从业者不可或缺的任务。为了满足不断增长的需求,许多人正在寻找可以帮助他们更高效地生成原创内容的解决方案。本文将专心分享一些优质的AI伪原创工具。 为什么要使用AI伪原创工具 随着内容创作的不断发展,越来…

爬虫学习日记第九篇(爬取seebug)

目标:https://www.seebug.org/vuldb/vulnerabilities 需求:爬取cve_id及影响组件 单线程 cookie是有时效的(过一段时间就不行了,大概半小时左右),但是并不需要登录(直接抓包拿到的请求头) import base64 import json import ur…

git操作:使用vscode集成

git操作方式 其实git操作一般有三种方式 分别是终端命令行,开发工具集成,专业的git可视化工具 我前面几章说的都是git的命令行操作,今天这篇文章主要是针对开发工具vscode集成git操作进行演示 说明一下,这里之所以选择vscode,是因为本人用的就是vscode,每个开发工具基本都有…

HNU计算机视觉作业二

前言 选修的是蔡mj老师的计算机视觉,上课还是不错的,但是OpenCV可能需要自己学才能完整把作业写出来。由于没有认真学,这门课最后混了80多分,所以下面作业解题过程均为自己写的,并不是标准答案,仅供参考 …

mysql支持的整数类型、各类型整数能够表示的数值范围

MySQL :: MySQL 8.2 Reference Manual :: 11.1.2 Integer Types (Exact Value) - INTEGER, INT, SMALLINT, TINYINT, MEDIUMINT, BIGINT mysql支持的整数有:TINYINT、SMALLINT、MEDIUMINT、INT(INT和INTEGER是同义词)、BIGINT,各…

自动化测试框架需要具备哪些功能?

我们经常听说使用了某某框架,那框架究竟是什么呢?框架有什么优势和功能? 什么是自动化框架 自动化框架是包含了自动化测试的组织、执行、监控以及报告等流程的工具,是由多个工具、库、模块和API等组成的工具集。自动化框架的目标…

致远互联FE协作办公平台 SQL注入漏洞复现

0x01 产品简介 致远互联FE协作办公平台是一款为企业提供全方位协同办公解决方案的产品。它集成了多个功能模块,旨在帮助企业实现高效的团队协作、信息共享和文档管理。 0x02 漏洞概述 致远互联FE协作办公平台 editflow_manager.js、validate.jsp等接口处存在SQL注入漏洞,未经…

Android studio生成二维码

1.遇到的问题 需要生成一个二维码&#xff0c;可以使用zxing第三方组件&#xff0c;增加依赖。 //生成二维码 implementation com.google.zxing:core:3.4.1 2.代码 展示页面 <ImageViewandroid:id"id/qrCodeImageView"android:layout_width"150dp"an…

Epoll服务器(ET工作模式)

目录 Epoll ET服务器设计思路Connection类TcpServer类 回调函数Accepter函数Recever函数Sender函数Excepter函数 事件处理套接字相关接口封装运行Epoll服务器 Epoll ET服务器 设计思路 在epoll ET服务器中&#xff0c;我们需要处理如下几种事件&#xff1a; 读事件&#xff…

如何恢复已删除的 JPG/JPEG 文件的方法深度解析!

您是否意外丢失或删除了 JPG 或 JPEG 照片&#xff1f;幸运的是&#xff0c;您可以使用照片恢复工具将它们恢复。立即获取适用于 PC 的 JPEG 恢复工具 - 照片恢复&#xff1a; 照片是捕捉和重温生活中特殊时刻的最佳方式。因此&#xff0c;当我们由于硬盘崩溃、意外格式化磁盘…

WordPress限制搜索关键词实现搜索黑名单

昨天有位站长问我能不能限制WordPress的搜索关键词&#xff0c;因为有人利用他的网站搜索色情词汇&#xff0c;本来正常搜索没有影响的&#xff0c;但是在部分网站中&#xff0c;搜索关键词产生的搜索页会被搜索引擎收录&#xff0c;实现推广功能。 WordPress的关键词搜索限制实…

kubesphere安装后启用DevOps

官方文档&#xff1a;KubeSphere DevOps 系统 1、集群管理---定制资源定义 进入目录&#xff1a;集群管理---定制资源定义搜索&#xff1a;clusterconfiguration 点击 ks-installer 右侧的 &#xff0c;选择编辑 YAML 在该 YAML 文件中&#xff0c;搜索 devops&#xff0c;…

k8s上安装KubeSphere

安装KubeSphere 前置环境安装nfs-server文件系统配置nfs-client配置默认存储创建了一个存储类metrics-server集群指标监控组件 安装KubeSphere执行安装查看安装进度 前置环境 下载配置我都是以CentOS 7.9 安装 k8s(详细教程)文章的服务器作为示例&#xff0c;请自行修改为自己的…

uniapp实战 —— 骨架屏

1. 自动生成骨架屏代码 在微信开发者工具中&#xff0c;预览界面点击生成骨架屏 确定后&#xff0c;会自动打开骨架屏代码文件 pages\index\index.skeleton.wxml 2. 将骨架屏代码转换为vue文件 在项目中新建文件 src\pages\index\components\skeleton.vue 将pages\index\index…