[c语言日寄]数据结构:栈

news2025/5/20 20:10:02

在这里插入图片描述

【作者主页】siy2333
【专栏介绍】⌈c语言日寄⌋:这是一个专注于C语言刷题的专栏,精选题目,搭配详细题解、拓展算法。从基础语法到复杂算法,题目涉及的知识点全面覆盖,助力你系统提升。无论你是初学者,还是进阶开发者,这里都能满足你的需求!
【食用方法】1.根据题目自行尝试 2.查看基础思路完善题解 3.学习拓展算法
【Gitee链接】资源保存在我的Gitee仓库:https://gitee.com/siy2333/study


文章目录

    • 前言:
    • 栈的基本概念
    • 栈的实现
      • 初始化
      • 入栈
      • 出栈
      • 查看栈顶元素
      • 检查栈是否为空
      • 获取栈的大小
      • 销毁栈
    • 栈的应用场景
      • 函数调用
      • 表达式求值
      • 括号匹配
      • 浏览器的前进和后退功能
    • 栈的优缺点
    • 栈的优化
      • 预分配内存
      • 使用循环缓冲区
    • 栈的测试
    • 完整代码
      • 头文件
      • 源文件
    • 栈的总结


前言:

在计算机科学中,数据结构是组织和存储数据的方式,而栈(Stack)是其中一种非常基础且重要的数据结构。它遵循后进先出(Last In First Out,LIFO)的原则,就像一叠盘子,你只能在最上面添加或移除盘子。本文将深入探讨栈的原理、实现方式以及应用场景,帮助读者更好地理解和运用这一数据结构。

文末附带完整的带有标准注释的c语言头文件以及源文件


栈的基本概念

栈是一种线性数据结构,它只允许在表的一端进行插入和删除操作。这一端被称为栈顶(Top),而另一端被称为栈底(Bottom)。栈的基本操作包括:

  1. 入栈(Push):将一个元素添加到栈顶。
  2. 出栈(Pop):移除栈顶的元素。
  3. 查看栈顶元素(Top):获取栈顶元素的值,但不移除它。
  4. 检查栈是否为空(Empty):判断栈中是否还有元素。
  5. 获取栈的大小(Size):返回栈中元素的数量。

这些操作的实现方式会因具体的编程语言和数据结构设计而有所不同,但基本原理是一致的。

栈的实现

栈可以通过数组或链表来实现。在本文中,我们将重点介绍基于数组的栈实现,因为它更简单且易于理解。以下是基于数组的栈实现的关键点:

初始化

在使用栈之前,需要对其进行初始化。这通常包括分配内存空间、设置栈顶指针和栈的容量。在我们的代码示例中,栈的初始化函数STInit会为栈分配初始容量为4的内存空间,并将栈顶指针设置为0。

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; // The next position of the current stack top element

    return;
}

入栈

当向栈中添加元素时,需要检查栈是否已满。如果栈已满,则需要扩容。扩容通常是通过分配更大的内存空间来实现的。在我们的代码中,当栈满时,会将栈的容量加倍。

void STPush(ST* ps, STDataType x)
{
    assert(ps);

    if (ps->capacity == ps->top)
    {
        STDataType* point = (STDataType*)realloc(ps->a, sizeof(STDataType) * ps->capacity * 2);
        if (point == NULL)
        {
            perror("realloc fail");
            return;
        }
        ps->a = point;
        ps->capacity *= 2;
    }

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

    return;
}

出栈

出栈操作相对简单,只需要将栈顶指针减1即可。但在出栈之前,需要检查栈是否为空,以避免出现错误。

void STPop(ST* ps)
{
    assert(ps);
    assert(!STEmpty(ps));

    ps->top--;

    return;
}

查看栈顶元素

查看栈顶元素的操作也很简单,只需要返回栈顶指针所指向的元素即可。同样,在执行此操作之前,需要检查栈是否为空。

STDataType STTop(ST* ps)
{
    assert(ps);
    assert(!STEmpty(ps));

    return ps->a[ps->top - 1];
}

检查栈是否为空

检查栈是否为空的操作可以通过比较栈顶指针是否为0来实现。

bool STEmpty(ST* ps)
{
    assert(ps);

    return ps->top == 0;
}

获取栈的大小

获取栈的大小的操作可以通过返回栈顶指针的值来实现。

int STSize(ST* ps)
{
    assert(ps);

    return ps->top;
}

销毁栈

在使用完栈后,需要对其进行销毁,以释放分配的内存空间。销毁栈的操作包括释放内存空间、将栈顶指针和容量设置为0。

void STDestory(ST* ps)
{
    assert(ps);

    free(ps->a);
    ps->a = NULL;
    ps->top = 0;
    ps->capacity = 0;

    return;
}

栈的应用场景

栈在计算机科学中有着广泛的应用,以下是一些常见的应用场景:

函数调用

在编程语言中,函数调用是通过栈来实现的。当一个函数被调用时,它的参数、返回地址和局部变量等信息会被压入调用栈中。当函数执行完毕后,这些信息会被弹出栈,程序会返回到调用函数的位置继续执行。

表达式求值

栈可以用于表达式的求值。例如,在计算后缀表达式时,可以使用一个栈来存储操作数。当遇到一个操作符时,从栈中弹出两个操作数,进行相应的运算,然后将结果压入栈中。当表达式计算完毕时,栈顶的元素就是表达式的值。

括号匹配

栈可以用于检查括号是否匹配。当遇到一个左括号时,将其压入栈中;当遇到一个右括号时,从栈中弹出一个左括号。如果在弹出左括号时栈为空,或者弹出的左括号与右括号不匹配,则说明括号不匹配。

浏览器的前进和后退功能

浏览器的前进和后退功能也可以通过栈来实现。当用户访问一个网页时,将该网页的地址压入一个栈中;当用户点击后退按钮时,从栈中弹出一个网页地址并跳转到该网页;当用户点击前进按钮时,将弹出的网页地址压入另一个栈中。

栈的优缺点

栈的优点包括:

  1. 简单易用:栈的实现相对简单,操作也容易理解。
  2. 高效:栈的基本操作(如入栈、出栈、查看栈顶元素等)的时间复杂度为O(1)。
  3. 适用范围广:栈在计算机科学中有着广泛的应用,如函数调用、表达式求值、回溯算法等。

然而,栈也有一些缺点:

  1. 容量限制:如果栈的容量有限,可能会出现栈溢出的情况。虽然可以通过动态扩容来解决这个问题,但动态扩容会增加时间和空间的开销。
  2. 只能在一端操作:栈只能在栈顶进行插入和删除操作,不能在其他位置进行操作。

栈的优化

虽然栈的基本操作已经非常高效,但在某些情况下,我们仍然可以通过一些优化手段来提高栈的性能。以下是一些常见的优化方法:

预分配内存

在初始化栈时,可以预分配一定量的内存空间,以减少动态扩容的次数。这可以提高栈的性能,但会增加内存的使用量。

使用循环缓冲区

循环缓冲区是一种特殊的数组结构,它可以循环使用内存空间。通过使用循环缓冲区,可以减少内存的分配和释放次数,从而提高栈的性能。

栈的测试

为了验证栈的正确性和性能,我们需要对其进行测试。以下是一个简单的测试程序,用于测试栈的基本操作:

#include"Stack.h"

void Test1()
{
    ST head;
    STInit(&head);

    STPush(&head, 1);
    STPush(&head, 2);
    STPush(&head, 3);
    STPush(&head, 4);
    STPush(&head, 5);

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

    return;
}

int main()
{
    Test1();

    return 0;
}

在这个测试程序中,我们首先初始化了一个栈,然后依次向栈中添加了5个元素。接着,我们依次从栈中弹出元素,并打印出每个元素的值。最后,我们销毁了栈。

完整代码

头文件

#pragma once

/**
* @file Stack.h
* @brief The header file of the Stack.
* @author siy2333
* @date 2025.5.11
*
* This header file defines the data structure and related operation functions of a Stack.
* Stack is a Last-In-First-Out (LIFO) data structure used for storing and managing data elements, 
	supporting operations to add (push) and remove (pop) elements at the top, 
	and is widely applied in program calling, expression evaluation, and other scenarios.
* Provides basic operations for adding, deleting, querying, and modifying.
*/

//Standard library header files included.
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<assert.h>

/**
* @brief The data types stored in the Stack.
*/
typedef int STDataType;

/**
* @typedef struct Stack.
* @brief Structure definition of Stack.
*/
typedef struct Stack
{
	STDataType* a;		///< The pointer to the memory for the Stack.
	int top;			///< The next position of the current Stack top element.
	int capacity;		///< The size of capacity.
}ST;

/**
* @brief Initialize Stack.
*/
void STInit(ST* ps);

/**
* @brief Destory the Stack.
*/
void STDestory(ST* ps);

/**
* @brief Store x in the top of the Stack.
*/
void STPush(ST* ps, STDataType x);

/**
* @brief Delete x at the top of the Stack.
*/
void STPop(ST* ps);

/**
* @brief return the capacity size of the Stack.
*/
int STSize(ST* ps);

/**
* @brief Check whether the Stack is empty.
* @return If the Stack is empty, return 1; Otherwise, return 0.
*/
bool STEmpty(ST* ps);

/**
* @brief Return the data at the top of the Stack.
* @return the data at the top of the Stack.
*/
STDataType STTop(ST* ps);

源文件

#include"Stack.h"

/**
* @details
	The faction use malloc to apply for 4 copies of memory for the Stack.
	The 0 will be store int the ps->top, it mean the next position of the current stack top element.
	The capacity will be set to 4.
* @warning ps mustn't be NULL.
*/
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;		//The next position of the current stack top element

	return;
}


/**
* @details 
	The faction will free the memory for the Stack;
	The top will be set to 0;
	The capacity will be set to 0.
* @warning ps mustn't be NULL.
*/
void STDestory(ST* ps)
{
	assert(ps);

	free(ps->a);
	ps->a = NULL;
	ps->top = 0;
	ps->capacity = 0;

	return;
}

/**
* @details 
	The faction will check whether the Stack is full;
	If the Stack is full, it will use realloc() to double the capacity, and refresh the ps->capacity.
	It will store the x in the ps->a[top], and refresh the ps->top.
* @warning ps mustn't be NULL.
*/
void STPush(ST* ps, STDataType x)
{
	assert(ps);
	
	if (ps->capacity == ps->top)
	{
		STDataType* point = (STDataType*)realloc(ps->a,sizeof(STDataType) * ps->capacity * 2);
		if (point == NULL)
		{
			perror("realloc fail");
			return;
		}
		ps->a = point;
		ps->capacity *= 2;
	}

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

	return;
}

/**
* @details 
	The function will refresh the ps->top.
* @warning ps musn't be NULL;
* @warning The Stack mustn't be empty.
*/
void STPop(ST* ps)
{
	assert(ps);
	assert(!STEmpty(ps));

	ps->top--;

	return;
}

/**
* @warning ps mustn't be NULL;
*/
int STSize(ST* ps)
{
	assert(ps);

	return ps->top;
}

/**
* @warning ps mustn't be NULL;
*/
bool STEmpty(ST* ps)
{
	assert(ps);

	return ps->top == 0;
}

/**
* @warning ps musn't be NULL;
* @warning The Stack mustn't be empty.
*/
STDataType STTop(ST* ps)
{
	assert(ps);
	assert(!STEmpty(ps));

	return ps->a[ps->top - 1];
}

栈的总结

栈是一种非常基础且重要的数据结构,它遵循后进先出的原则。栈的基本操作包括入栈、出栈、查看栈顶元素、检查栈是否为空和获取栈的大小等。栈可以通过数组或链表来实现,其中基于数组的栈实现相对简单且易于理解。栈在计算机科学中有着广泛的应用,如函数调用、表达式求值、回溯算法等。虽然栈有一些缺点,但它的优点使其在许多场景下都非常有用。通过一些优化手段,我们可以进一步提高栈的性能。总之,栈是一种非常重要的数据结构,值得我们深入学习和研究。

希望本文对您理解栈有所帮助。如果您有任何问题或建议,欢迎在评论区留言讨论。

[专栏链接QwQ] :⌈c语言日寄⌋CSDN
[关注博主ava]:siy2333
感谢观看~ 我们下次再见!!

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

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

相关文章

WEB安全--Java安全--LazyMap_CC1利用链

一、前言 该篇是基于WEB安全--Java安全--CC1利用链-CSDN博客的补充&#xff0c;上篇文章利用的是TransformedMap类&#xff0c;而CC链的原作者是利用的LazyMap类作为介质进行的触发。 所以本文将分析国外原作者在ysoserial commonscollections1中给出的CC1利用链。 二、回顾梳…

黑马k8s(六)

1.Deployment&#xff08;Pod控制器&#xff09; Selector runnginx 标签选择&#xff1a;会找pod打的标签 执行删除之后&#xff0c;pod也会删除&#xff0c;Terminating正在删除 如果想要访问其中的一个pod借助&#xff1a;IP地址端口号访问 假设在某一个瞬间&#xff0c;…

【OpenGL学习】(一)创建窗口

文章目录 【OpenGL学习】&#xff08;一&#xff09;创建窗口 【OpenGL学习】&#xff08;一&#xff09;创建窗口 GLFW OpenGL 本身只是一套图形渲染 API&#xff0c;不提供窗口创建、上下文管理或输入处理的功能。 GLFW 是一个支持创建窗口、处理键盘鼠标输入和管理 OpenGL…

AI大语言模型评测体系演进与未来展望

随着人工智能技术的飞速发展,大语言模型(LLMs)已成为自然语言处理领域的核心研究方向。2025年最新行业报告显示,当前主流模型的评测体系已从单一任务评估转向多维度、全链路的能力剖析。例如,《全球首个大语言模型意识水平”识商”白盒DIKWP测评报告》通过数据、信息、知识…

微服务项目->在线oj系统(Java版 - 5)

相信自己,终会成功 微服务代码: lyyy-oj: 微服务 目录 C端代码 用户题目接口 修改后用户提交代码(应用版) 用户提交题目判题结果 代码沙箱 1. 代码沙箱的核心功能 2. 常见的代码沙箱实现方式 3. 代码沙箱的关键问题与解决方案 4. 你的代码如何与沙箱交互&#xff1f; …

get请求使用数组进行传参

get请求使用数组进行传参,无需添加中括号 mvc接口要添加参数名&#xff0c;使用array承接。不能用list, 否则会报错 这里是用apifox模拟前端调用。 前端调用代码 // 根据项目ID和角色ID查询相关审批人 export function findRelativeApproverByProjectIdAndRoleId(roleIds, p…

【MySQL成神之路】MySQL常用语法总结

目录 MySQL 语法总结 数据库操作 表操作 数据操作 查询语句 索引操作 约束 事务控制 视图操作 存储过程和函数 触发器 用户和权限管理 数据库操作 创建数据库&#xff1a; CREATE DATABASE database_name; 选择数据库&#xff1a; USE database_name; 删除数…

Linux动静态库制作与原理

什么是库 库是写好的现有的&#xff0c;成熟的&#xff0c;可以复用的代码。现实中每个程序都要依赖很多基础的底层库&#xff0c;不可能每个人的代码都从零开始&#xff0c;因此库的存在意义非同寻常。 本质上来说库是一种可执行代码的二进制形式&#xff0c;可以被操作系统…

ffmpeg 把一个视频复制3次

1. 起因&#xff0c; 目的: 前面我写过&#xff0c;使用 python 把一个视频复制3次但是速度太慢了&#xff0c;我想试试看能否改进。而且我想换一种新的视频处理思路&#xff0c;并试试看速度如何。 2. 先看效果 效果就是能行&#xff0c;而且速度也快。 3. 过程: 代码 1…

GPT/Claude3国内免费镜像站更新 亲测可用

无限次使用&#xff1a;无限制的提问次数&#xff0c;不设上限&#xff0c;随心所欲。 无需魔法、稳定流畅&#xff1a;操作简便&#xff0c;无需复杂设置&#xff0c;即可享受稳定流畅的服务。 手机和电脑均能用&#xff1a;轻松适配手机和电脑&#xff0c;使用体验更佳。 …

Python:操作Excel按行写入

Python按行写入Excel数据,5种实用方法大揭秘! 在日常的数据处理和分析工作中,我们经常需要将数据写入到Excel文件中。Python作为一门强大的编程语言,提供了多种库和方法来实现将数据按行写入Excel文件的功能。本文将详细介绍5种常见的Python按行写入Excel数据的方法,并附上…

Redis进阶知识

Redis 1.事务2. 主从复制2.1 如何启动多个Redis服务器2.2 监控主从节点的状态2.3 断开主从复制关系2.4 额外注意2.5拓扑结构2.6 复制过程2.6.1 数据同步 3.哨兵选举原理注意事项 4.集群4.1 数据分片算法4.2 故障检测 5. 缓存5.1 缓存问题 6. 分布式锁 1.事务 Redis的事务只能保…

12.vue整合springboot首页显示数据库表-实现按钮:【添加修改删除查询】

vue整合springboot首页显示数据库表&#xff1a;【添加修改删除查询】 提示&#xff1a;帮帮志会陆续更新非常多的IT技术知识&#xff0c;希望分享的内容对您有用。本章分享的是node.js和vue的使用。前后每一小节的内容是存在的有&#xff1a;学习and理解的关联性。【帮帮志系…

bisheng系列(一)- 本地部署(Docker)

目录 一、导读 二、说明 1、镜像说明 2、本节内容 三、docker部署 1、克隆代码 2、运行镜像 3、可能的错误信息 四、页面测试 1、注册用户 2、登陆成功 3、添加模型 一、导读 环境&#xff1a;Ubuntu 24.04、Windows 11、WSL 2、Python 3.10 、bisheng 1.1.1 背景…

如何用Python批量解压ZIP文件?快速解决方案

如何用Python批量解压ZIP文件&#xff1f;快速解决方案 文章目录 **如何用Python批量解压ZIP文件&#xff1f;快速解决方案**代码结果详细解释 话不多说&#xff0c;先上干货&#xff01;&#xff01;&#xff01; 代码 import os import zipfiledef unzip_file(dir_path: str…

DriveGenVLM:基于视觉-语言模型的自动驾驶真实世界视频生成

《DriveGenVLM: Real-world Video Generation for Vision Language Model based Autonomous Driving》2024年8月发表&#xff0c;来自哥伦比亚大学的论文。 自动驾驶技术的进步需要越来越复杂的方法来理解和预测现实世界的场景。视觉语言模型&#xff08;VLM&#xff09;正在成…

企业标准信息公共服务平台已开放标准通编辑器访问入口

标准通 数字化标准编辑器 专业、高效、便捷 企业标准信息公共服务平台 近日&#xff0c;企业标准信息公共服务平台已开放标准通编辑器访问入口&#xff0c;可进入官网指定版块使用&#xff01; 核心功能亮点 解决企业痛点 传统标准编制&#xff0c;需反复核对格式、逐条…

进阶-数据结构部分:1、数据结构入门

飞书文档https://x509p6c8to.feishu.cn/wiki/HRLkwznHiiOgZqkqhLrcZNqVnLd 一、存储结构 顺序存储 链式存储 二、常用数据结构 2.1、栈 先进后出 场景&#xff1a; 后退/前进功能&#xff1a;网页浏览器中的后退和前进按钮可以使用栈来实现。在浏览网页时&#xff0c;每次…

React 19中useContext不需要Provider了。

文章目录 前言一、React 19中useContext移除了Provider&#xff1f;二、使用步骤总结 前言 在 React 19 中&#xff0c;useContext 的使用方式有所更新。开发者现在可以直接使用 作为提供者&#xff0c;而不再需要使用 <Context.Provider>。这一变化简化了代码结构&…

Json schema校验json字符串(networknt/json-schema-validator库)

学习链接 json-schema官网 - 英文 jsonschemavalidator 可在线校验网站 networknt的json-schema-validator github地址 networknt的json-schema-validator 个人gitee地址 - 里面有md文档说明和代码示例 JSON Schema 入门指南&#xff1a;如何定义和验证 JSON 数据结构 JS…