当前位置: 首页 > news >正文

【数据结构】栈详解

目录

  • 1. 前言
  • 2. 栈
    • 2.1 栈的概念及结构
    • 2.2 如何实现栈
    • 2.3 数组栈实现
      • 2.3.1 top怎么确定
      • 2.3.2 栈顶插入
        • 2.3.2.1 栈顶插入分析
        • 2.3.2.2 栈顶插入代码实现
      • 2.3.3 栈顶删除
      • 2.3.4 判空
        • 2.3.4.1 分析
        • 2.3.4.2 代码实现
      • 2.3.5 栈的元素个数
      • 2.3.6 栈销毁
      • 2.3.7 栈访问数据
  • 3. 源代码
    • 3.1 Stack.h
    • 3.2 Stack.c
    • 3.3 test.c

1. 前言

在前面我们一起了解的数据结构有顺序表和链表,这次来介绍栈。
与顺序表和链表相同的是,栈也是常见的数据结构。而与前面两种不同的是,它在农村种的存储,接下来让我们一起来学习一下。

2. 栈

2.1 栈的概念及结构

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

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

出栈:栈的删除操作叫做出栈。出数据也在栈顶。
在这里插入图片描述

2.2 如何实现栈

在这里插入图片描述
那该如何实现栈呢?
第一种使用数组栈
第二种使用链式栈

链表实现又分为双向链表,和单链表。
双向链表实现,栈顶可以是尾,也可以是头。
单链表实现,栈顶只能是头。
在这里插入图片描述
如果只选择一种来实现,那必然是数组,虽然有扩容,但不是频繁扩容。还有另外一个优势,它访问数据,CPU高速缓存命中率比较高,访问第一个,后面都在缓存了。

2.3 数组栈实现

2.3.1 top怎么确定

我们需要考虑top怎么确定?
如果top给0,那么表示的是栈顶还是栈顶位置的下一个?
在这里插入图片描述

如果空的时候top给的是0,那么插入一个数据之后top也是0,因为top指向栈顶。
此时就出现了歧义。top==0是一个元素还是空?区分不开。

那该怎么区分呢?
第一种如果top指向栈顶元素,那么top初始时给-1。

在这里插入图片描述
这种插入数据时:
要先加加top,再在这个位置赋值。
在这里插入图片描述

第二种指向栈顶元素的下一个。
此时top初始时就为0。
在这里插入图片描述
这种插入数据时:
要先在这个位置赋值,再加加top。
在这里插入图片描述

这里插入就取决于怎么初始化栈。

选择第二种来实现代码,这种更简单。

void STInit(ST* pst)
{assert(pst);pst->a = NULL;pst->capacity = 0;pst->top = 0;//第二种
}

2.3.2 栈顶插入

2.3.2.1 栈顶插入分析

插入的代码非常简单:

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

但是在插入之前,要先判断一下栈有没有满,满的化要进行扩容。

if (pst->top == pst->capacity)

但我们要初始时候有没有给空间,如果有直接扩两倍,没有就给初始值4。
使用一个三目表达式

int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;

如果没有开空间,就直接返回或者结束程序。

if (tmp == NULL){perror("realloc fail");return;}
2.3.2.2 栈顶插入代码实现
void STPush(ST* pst, STDataType x)
{assert(pst);if (pst->top == pst->capacity){int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;STDataType* tmp = (STDataType*)realloc(pst->a, sizeof(STDataType) * newcapacity);if (tmp == NULL){perror("realloc fail");return;}pst->a = tmp;pst->capacity = newcapacity;}pst->a[pst->top] = x;pst->top++;
}

2.3.3 栈顶删除

删除很简单,直接top–,就行,但减减之前判断一下top是不是0,加一个断言就行。
代码实现:

void STPop(ST* pst)
{assert(pst);// 不为空assert(pst->top > 0);pst->top--;
}

返回栈顶数据的代码:

STDataType STTop(ST* pst)
{assert(pst);// 不为空assert(pst->top > 0);return pst->a[pst->top - 1];
}

2.3.4 判空

2.3.4.1 分析

判空这里得看刚开始时:初始时top为0,就要判断top==0。初始为-1,那么就要判断top==-1
可以用if来判断:

if (pst->top == 0){return true;}else{return false;}

但是有个更简单的:bool类型返回就是真假,利用返回值当结果就行,直接返回pst->top == 0

2.3.4.2 代码实现

代码实现:

bool STEmpty(ST* pst)
{assert(pst);/*if (pst->top == 0){return true;}else{return false;}*/return pst->top == 0;
}

2.3.5 栈的元素个数

如果top是指向栈顶元素的下一个,那么元素个数就是top。
如果top就是指向栈顶,那么元素个数就是top+1。
在这里插入图片描述

代码实现:

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

2.3.6 栈销毁

直接将元素都free掉,再把它们都置为空。把栈顶和栈空间都置为0。

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

2.3.7 栈访问数据

因为栈是后进先出,所以栈不能随便打印。
那怎么打印呢?
判断栈是否为空,然后从栈顶开始访问。
访问了栈顶元素,要想访问下一个就要先将栈顶元素弹出,直到栈为空,就结束。

代码实现:

while (!STEmpty(&s)){printf("%d ", STTop(&s));STPop(&s);}printf("\n");

3. 源代码

3.1 Stack.h

#pragma once
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<stdbool.h>typedef int STDataType;typedef struct Stack
{int* a;int top;		// 标识栈顶位置的int capacity;
}ST;void STInit(ST* pst);
void STDestroy(ST* pst);// 栈顶插入删除
void STPush(ST* pst, STDataType x);
void STPop(ST* pst);
STDataType STTop(ST* pst);bool STEmpty(ST* pst);
int STSize(ST* pst);

3.2 Stack.c

#include"Stack.h"void STInit(ST* pst)
{assert(pst);pst->a = NULL;pst->capacity = 0;// 表示top指向栈顶元素的下一个位置pst->top = 0;// 表示top指向栈顶元素//pst->top = -1;
}void STDestroy(ST* pst)
{assert(pst);free(pst->a);pst->a = NULL;pst->top = pst->capacity = 0;
}// 栈顶插入删除
void STPush(ST* pst, STDataType x)
{assert(pst);if (pst->top == pst->capacity){int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;STDataType* tmp = (STDataType*)realloc(pst->a, sizeof(STDataType) * newcapacity);if (tmp == NULL){perror("realloc fail");return;}pst->a = tmp;pst->capacity = newcapacity;}pst->a[pst->top] = x;pst->top++;
}void STPop(ST* pst)
{assert(pst);// 不为空assert(pst->top > 0);pst->top--;
}STDataType STTop(ST* pst)
{assert(pst);// 不为空assert(pst->top > 0);return pst->a[pst->top - 1];
}bool STEmpty(ST* pst)
{assert(pst);/*if (pst->top == 0){return true;}else{return false;}*/return pst->top == 0;
}int STSize(ST* pst)
{assert(pst);return pst->top;
}

3.3 test.c

#include"Stack.h"int main()
{ST s;STInit(&s);STPush(&s, 1);STPush(&s, 2);STPush(&s, 3);printf("%d ", STTop(&s));STPop(&s);printf("%d ", STTop(&s));STPop(&s);STPush(&s, 4);STPush(&s, 5);//    一     对     多// 入栈顺序  --  出栈顺序while (!STEmpty(&s)){printf("%d ", STTop(&s));STPop(&s);}printf("\n");return 0;
}

相关文章:

【数据结构】栈详解

目录 1. 前言2. 栈2.1 栈的概念及结构2.2 如何实现栈2.3 数组栈实现2.3.1 top怎么确定2.3.2 栈顶插入2.3.2.1 栈顶插入分析2.3.2.2 栈顶插入代码实现 2.3.3 栈顶删除2.3.4 判空2.3.4.1 分析2.3.4.2 代码实现 2.3.5 栈的元素个数2.3.6 栈销毁2.3.7 栈访问数据 3. 源代码3.1 Stac…...

大结局!OpenAI创始人奥特曼和 Greg Brockman 将加入微软!!!

持续48小时的OpenAI政变大戏终于迎来了大结局&#xff01; 微软堪称最大赢家&#x1f4a5;&#x1f4a5;&#x1f4a5; 微软CEO刚刚宣布&#xff1a; 我们仍然致力于与 OpenAI 的合作伙伴关系&#xff0c;并对我们的产品路线图、我们在 Microsoft Ignite 上宣布的一切继续创…...

Linux QT交叉编译环境安装

参考链接 linux交叉编译Qt_linux qt 交叉编译-CSDN博客 关键点&#xff1a;编译脚本&#xff0c;放在qt源代码根目录的.sh文件 #!/bin/shcd ./qt-everywhere-src-5.12.9./configure -prefix /home/qsqya/compile/qt5.12.9/build \ -opensource \ -release \ -confirm-license…...

媲美有线操作,支持4KHz响应和无线充电的游戏鼠标,雷柏VT3S上手

对于无线鼠标来说&#xff0c;操作延迟和精度对游戏操作影响很大&#xff0c;常见的游戏鼠标至少都有1KHz的回报率&#xff0c;而雷柏今年已经出了很多支持4KHz回报的鼠标了&#xff0c;像是我现在用的这款VT3S游戏鼠标&#xff0c;就搭载了旗舰级的原相3395引擎&#xff0c;支…...

【Flask使用】全知识md文档,4大部分60页第3篇:状态cookie和session保持

本文的主要内容&#xff1a;flask视图&路由、虚拟环境安装、路由各种定义、状态保持、cookie、session、模板基本使用、过滤器&自定义过滤器、模板代码复用&#xff1a;宏、继承/包含、模板中特有变量和函数、Flask-WTF 表单、CSRF、数据库操作、ORM、Flask-SQLAlchemy…...

类方法,静态方法和实例方法的区别及应用场景

在 Python 中&#xff0c;有三种不同类型的方法&#xff1a;实例方法、类方法和静态方法。它们各自有不同的特点和应用场景&#xff1a; 实例方法&#xff08;Instance Method&#xff09;&#xff1a; 实例方法是最常见的方法类型&#xff0c;在方法定义中第一个参数通常被命…...

CleanMyMac X4.16免费版mac电脑一键清理电脑垃圾工具

但是&#xff0c;我最近发现随着使用时间的增加&#xff0c;一些奇奇怪怪的文件开始占据有限的磁盘空间&#xff0c;存储空间变得越来越小&#xff0c;系统占用空间越来越大&#xff0c;越来越多的无效文件开始影响我电脑的运行速度。 Mac的文件管理方式和Windows不太一样&…...

汽车级低压差稳压器LDO LM317BD2TR4G原理、参数及应用

LM317BD2TR4G主要功能特性分析 &#xff1a; LM317BD2TR4G 低漏 (LDO) 线性电压稳压器是一款可调 3 端子正向 LDO 电压器&#xff0c;能够在 1.2 V 至 37 V 的输出电压范围内提供 1.5 A 以上的电流。此电压稳压器使用非常简便&#xff0c;仅需两个外部电阻即可设置输出电压。另…...

多对多的创建方式与Ajax

模型层补充 MTV与MVC模型 MTV 全称 Models Templates Views 模型模板视图 MVC 全称 Models Views Controller 模型视图控制MTV: Django号称是MTV模型 MVC: 其实django本质也是MVC 拓展: vue框架:MVVM模型choices参数(数据库字段设计常见) choices使用 class User(models.Mod…...

【Linux网络】详解使用http和ftp搭建yum仓库,以及yum网络源优化

目录 一、回顾yum的原理 1.1yum简介 yum安装的底层原理&#xff1a; yum的好处&#xff1a; 二、学习yum的配置文件及命令 1、yum的配置文件 2、yum的相关命令详解 3、yum的命令相关案例 三、搭建yum仓库的方式 1、本地yum仓库建立 2、通过http搭建内网的yum仓库 3、…...

算法设计与分析算法实现——动态规划最大子段

输入&#xff1a;整数序列a1,a2,…,an 输出&#xff1a;序列的一个子段&#xff0c;其和Σak最大 注意&#xff1a;当所有整数都为负数时&#xff0c;定义最大子段和为0 使用动态规划&#xff0c;输入数组是a[n]&#xff1b; 状态转移方程dp[i]max(dp[i-1]a[i],a[i])——这个状…...

JavaWeb-JVM内存管理机制

JavaWeb-JVM内存管理机制 一、JVM内存管理概述1.1 什么是JVM内存管理1.2 物理内存与虚拟内存1.3 内核空间与用户空间二、java中哪些组建需要使用内存2.1 Java堆2.2 线程2.3 类和类加速器2.4 NIO2.5 JNI三、JVM内存结构3.1 PC寄存器3.2 Java栈3.3 堆3.4 方法区3.5 运行时常量池3…...

阿里云oss存储文件上传功能实现(保姆级教程)

先登录&#xff1a; 点击进入控制台 点击左上角导航栏按钮 搜索oss&#xff0c;点击进入 进入之后点击立即开通oss按钮&#xff0c;开通之后点击下图立即创建&#xff0c;弹出创建Bucket 填上Bucket名称&#xff0c;读写权限改为公共读。其他不变点击确定创建&#xff0c;完成…...

centos7配置 局域网自动解析hostname

这样可以让局域网别的电脑直接通过hostname来连接这台电脑。 如果不是windows系统&#xff0c;可以用hostname.local来连接 主要是用到了mdns的功能&#xff0c;需要安装nss-mdns。 vmware下nat模式下&#xff0c;宿主机也可以通过连接hostname使用。 yum install epel-releas…...

wireshark 过滤设置

gpt: Wireshark 是一个网络分析工具&#xff0c;可以用来捕获和分析网络数据包。你可以使用过滤器来筛选并查看你感兴趣的数据包。Wireshark 使用的是基于BPF&#xff08;Berkeley Packet Filter&#xff09;语法的过滤器。以下是一些常见的 Wireshark 过滤器设置&#xff1a;…...

SpringBoot-过滤器Filter+JWT令牌实现登录验证

登录校验-Filter 分析 过滤器Filter的快速入门以及使用细节我们已经介绍完了&#xff0c;接下来最后一步&#xff0c;我们需要使用过滤器Filter来完成案例当中的登录校验功能。 我们先来回顾下前面分析过的登录校验的基本流程&#xff1a; 要进入到后台管理系统&#xff0c;我…...

VMware——WindowServer2012R2环境安装mysql5.7.14解压版_互为主从(图解版)

目录 一、服务器信息二、192.168.132.35服务器上安装mysql&#xff08;主&#xff09;2.1、环境变量配置2.2、安装2.2.1、修改配置文件内容2.2.2、初始化mysql并指定超级用户密码2.2.3、安装mysql服务2.2.4、启动mysql服务2.2.5、登录用户管理及密码修改2.2.6、开启远程访问 三…...

python 实现蚁群算法(simpy带绘图)

这里使用了蚁群算法求解了旅行商问题&#xff0c;同时结合了simpy来绘图 选择下一个食物的函数为&#xff1a; probability[i] pheromone[self.now][self.not_to_foods[i]] ** pheromone_w (1 / distance[self.now][self.not_to_foods[i]]) ** distance_w 该条路概率权重该点…...

OpenAI 董事会宫斗始作俑者?一窥伊尔亚·苏茨克维内心世界

OpenAI 董事会闹剧应该是暂告一个段落了,Sam Altman和Greg Brockman等一众高管均已加入微软,还有员工写联名信逼宫董事会的戏码,关注度已经降下来了。 但是,这场宫斗闹剧的中心人物Ilya Sutskever大家关注度不算太高。他本人是纯粹的技术男,极少抛头露面透露其内心世界。…...

Android App 启动状态有几种?

startup state Android 启动状态&#xff08;startup state&#xff09;1.1、冷启动&#xff08;Cold Start&#xff09;1.2、温启动&#xff08;Warm Start&#xff09;1.3、热启动&#xff08;Hot Start&#xff09;1.4、后台启动&#xff08;Background Start&#xff09; 优…...

3分钟配置Spyder深色模式:Python开发者的护眼终极指南

3分钟配置Spyder深色模式&#xff1a;Python开发者的护眼终极指南 【免费下载链接】spyder Official repository for Spyder - The Scientific Python Development Environment 项目地址: https://gitcode.com/gh_mirrors/sp/spyder Spyder作为Python科学计算的强大IDE&…...

SI5351高频PCB布局避坑指南:从200MHz信号完整性问题到实测波形分析

SI5351高频PCB布局避坑指南&#xff1a;从200MHz信号完整性问题到实测波形分析 在射频电路设计中&#xff0c;时钟信号的纯净度往往决定着整个系统的性能上限。SI5351作为一款支持8通道输出的可编程时钟发生器&#xff0c;其200MHz的输出能力既带来了设计灵活性&#xff0c;也带…...

雷总发福利了!小米100万亿Token免费领,还没上车的速进!

搞AI、敲代码、或者平时爱折腾AI大模型的朋友注意了。 最近小米开源了自家的旗舰大模型 MiMo-V2.5 系列&#xff0c;不仅把支持100万上下文窗口的模型直接开源&#xff0c;还顺手整了个大活——推出了个叫“MiMo Orbit 百万亿 Token 创造者激励计划”的活动。 大白话翻译过来就…...

Rust构建跨平台AI桌面应用:PoleStar Chat的多机器人协同与本地化实践

1. 项目概述&#xff1a;一个用Rust重写的跨平台AI聊天桌面应用如果你和我一样&#xff0c;每天的工作流里离不开ChatGPT、Claude或者Gemini&#xff0c;那你肯定也受够了在浏览器标签页之间来回切换&#xff0c;或者忍受着某些官方客户端那捉襟见肘的功能和时不时卡顿的体验。…...

【PAT甲级真题】- Elevator(20)

题目来源 Elevator 题目描述点击链接自行查看 注意点&#xff1a; 停在同一层时多等5秒 Description The highest building in our city has only one elevator. A request list is made up with N positive numbers. The numbers denote at which floors the elevator will…...

从惠普档案火灾看电子测试测量技术遗产的保护与传承

1. 一场大火与一段历史的消逝&#xff1a;从惠普档案损毁看技术遗产的脆弱性2017年10月&#xff0c;加州葡萄酒乡那场被称为“塔布斯”的山火&#xff0c;不仅吞噬了无数家园与生命&#xff0c;也在不经意间&#xff0c;灼伤了现代电子工程史的一角。当烈焰席卷位于圣罗莎的是德…...

NanoPi M6硬件解析与嵌入式开发实践

1. NanoPi M6 硬件架构深度解析NanoPi M6 是一款基于 Rockchip RK3588S SoC 设计的单板计算机&#xff0c;其硬件配置在当前 SBC 领域堪称旗舰级。作为长期从事嵌入式开发的工程师&#xff0c;我认为这款板卡最值得关注的是其平衡的性能与扩展性设计。1.1 核心处理器性能剖析RK…...

WinCC组态没问题,数据就是存不进U盘?手把手教你诊断西门子触摸屏USB接口‘假死’

WinCC组态正确却无法存储数据&#xff1f;深度解析西门子触摸屏USB接口故障排查 最近在工业自动化论坛上&#xff0c;看到不少工程师反馈一个奇怪现象&#xff1a;明明WinCC组态完全正确&#xff0c;数据记录配置也没问题&#xff0c;但就是无法将数据存入U盘。这种"组态正…...

如何使用Casbin RBAC域API实现多租户角色权限管理:完整指南

如何使用Casbin RBAC域API实现多租户角色权限管理&#xff1a;完整指南 【免费下载链接】casbin Apache Casbin: an authorization library that supports access control models like ACL, RBAC, ABAC. 项目地址: https://gitcode.com/GitHub_Trending/ca/casbin 在现代…...

别再乱起名了!Windows文件命名避坑指南:从CON到260字符限制,这些坑你踩过吗?

Windows文件命名避坑实战&#xff1a;从CON到长路径的终极解决方案 你是否曾在命令行中尝试创建名为CON.txt的文件却遭遇系统拒绝&#xff1f;或是将精心整理的文档同步到云端时&#xff0c;突然提示"路径过长无法传输"&#xff1f;这些看似简单的文件命名问题&#…...