数据结构中的栈(C语言版)

一.栈的概念

栈是一种常见的数据结构,它遵循后进先出的原则。栈可以看作是一种容器,其中的元素按照一种特定的顺序进行插入和删除操作。

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

1.栈的特点

    1.元素的插入和删除操作只能在栈的一端进行,该端被称为栈顶。

    2.最后插入的元素是第一个被删除的元素,因此称为后进先出。

    3.栈中的元素没有编号或索引,只有栈顶指针来指示栈的当前位置。

2.栈的优点

  1. 简单高效:栈的操作是基于后进先出(LIFO)的原则,入栈和出栈操作都只涉及栈顶元素,因此操作的时间复杂度都是O(1),使得栈的操作非常高效。

  2. 空间效率高:栈的底层实现可以使用数组或链表,无论是使用静态数组还是动态链表,都可以根据实际需要灵活分配内存,因此在空间利用上比较高效。

  3. 递归和回溯:栈在递归和回溯算法中扮演着重要的角色。递归函数调用时会将当前函数的状态(包括局部变量、返回地址等)压入栈中,当递归函数返回时,栈顶的状态会被弹出,恢复到上一层递归函数的状态。

  4. 撤销操作:栈可以用于实现撤销操作,比如文本编辑器中的撤销功能。每当执行一个操作时,将操作的状态存储在栈中,当需要撤销时,只需从栈中弹出最近的状态。

3.栈的缺点

  1. 容量限制:栈的容量是有限的,无论是基于数组还是链表实现的栈,都会受到内存大小的限制。当栈的元素个数超过容量时,会发生栈上溢(stack overflow)的错误。

  2. 无随机访问:栈的特点是只能在栈顶进行插入和删除操作,没有提供随机访问的能力。如果需要访问或修改栈中的其他元素,必须先将栈顶的元素弹出,直到达到目标位置。

  3. 不灵活:栈的特性决定了它的使用场景受到一定的限制。对于需要随机访问、频繁插入和删除的场景,栈可能不是最佳选择。

二.栈的功能

栈作为一种数据结构,具有以下几个主要的功能:

  1. 入栈:将元素添加到栈的顶部(栈顶)。新元素成为栈顶,原有的栈顶元素依次向下移动。入栈操作可以用于将数据添加到栈中。

  2. 出栈:从栈的顶部(栈顶)移除元素。被移除的元素是最后一个入栈的元素,即栈顶元素。出栈操作会改变栈的结构,并返回被移除的元素。

  3. 获取栈顶元素:获取栈顶的元素,但不对栈进行修改。这个操作可以让我们查看栈顶的元素,而不改变栈的结构。

  4. 判断栈是否为空:检查栈是否不包含任何元素。如果栈中没有元素,即栈为空,该函数返回真;否则,返回假。

  5. 判断栈是否已满:检查栈是否已达到其容量上限。对于基于数组实现的栈,如果数组已满,即栈已满,该函数返回真;否则,返回假。

三.栈的实现

1.创建栈

创建一个结构体,里面的成员是数组以及指针。(在这里,为了大家能够方便理解,用静态的顺序表来实现)

#include <stdio.h>
#define MAX_SIZE 100

typedef struct {
    int data[MAX_SIZE]; // 用于存储栈中的元素
    int top; // 栈顶指针,指向栈顶元素的索引
} Stack;

2.初始化栈

将栈顶的指针初始化为-1,表示此栈为空。

// 初始化栈
void initStack(Stack* stack) {
    stack->top = -1; // 栈顶指针初始化为-1,表示栈为空
}

3.判断栈是否为空

判断栈是否为空。如果栈顶指针top等于-1,表示栈为空,返回1;否则,返回0。

// 判断栈是否为空
int isEmpty(Stack* stack) {
    return stack->top == -1; // 栈为空时,栈顶指针为-1
}

4. 判断是否已满

判断栈是否已满。如果栈顶指针top等于数组最大索引(MAX_SIZE - 1),表示栈已满,返回1;否则,返回0。

// 判断栈是否已满
int isFull(Stack* stack) {
return stack->top == MAX_SIZE - 1; // 栈满时,栈顶指针等于数组最大索引
}

5.入栈

 入栈操作。首先使用isFull函数检查栈是否已满,如果已满,则打印错误信息并返回;否则,将栈顶指针top加1,并将元素item放入栈顶位置data[top]

// 入栈
void push(Stack* stack, int item) {
if (isFull(stack)) {
printf("Stack overflow!\n"); // 栈已满,无法入栈
return;
}
stack->data[++stack->top] = item; // 栈顶指针加1,并将元素放入栈顶
}

 6.出栈

出栈操作。首先使用isEmpty函数检查栈是否为空,如果为空,则打印错误信息并返回-1;否则,返回栈顶元素data[top],并将栈顶指针top减1。

// 出栈
int pop(Stack* stack) {
if (isEmpty(stack)) {
printf("Stack underflow!\n"); // 栈为空,无法出栈
return -1;
}
return stack->data[stack->top--]; // 返回栈顶元素,并将栈顶指针减1
}

 7.获取栈顶元素

获取栈顶元素。首先使用isEmpty函数检查栈是否为空,如果为空,则打印错误信息并返回-1;否则,返回栈顶元素data[top],但不修改栈的结构。

/ 获取栈顶元素
int peek(Stack* stack) {
if (isEmpty(stack)) {
printf("Stack is empty!\n"); // 栈为空,无栈顶元素
return -1;
}
return stack->data[stack->top]; // 返回栈顶元素,不修改栈的结构
}

8.打印栈中元素

 打印栈中的元素。首先使用isEmpty函数检查栈是否为空,如果为空,则打印提示信息;否则,使用循环从栈底到栈顶依次打印栈中的元素。

/ 打印栈中的元素(用于调试)
void printStack(Stack* stack) {
if (isEmpty(stack)) {
printf("Stack is empty!\n");
return;
}
printf("Stack: ");
for (int i = 0; i <= stack->top; i++) {
printf("%d ", stack->data[i]);
}
printf("\n");
}

四.栈的源码呈现

 

#include <stdio.h>
#define MAX_SIZE 100

typedef struct {
    int data[MAX_SIZE]; // 用于存储栈中的元素
    int top; // 栈顶指针,指向栈顶元素的索引
} Stack;

// 初始化栈
void initStack(Stack* stack) {
    stack->top = -1; // 栈顶指针初始化为-1,表示栈为空
}

// 判断栈是否为空
int isEmpty(Stack* stack) {
    return stack->top == -1; // 栈为空时,栈顶指针为-1
}

// 判断栈是否已满
int isFull(Stack* stack) {
    return stack->top == MAX_SIZE - 1; // 栈满时,栈顶指针等于数组最大索引
}

// 入栈
void push(Stack* stack, int item) {
    if (isFull(stack)) {
        printf("Stack overflow!\n"); // 栈已满,无法入栈
        return;
    }
    stack->data[++stack->top] = item; // 栈顶指针加1,并将元素放入栈顶
}

// 出栈
int pop(Stack* stack) {
    if (isEmpty(stack)) {
        printf("Stack underflow!\n"); // 栈为空,无法出栈
        return -1;
    }
    return stack->data[stack->top--]; // 返回栈顶元素,并将栈顶指针减1
}

// 获取栈顶元素
int peek(Stack* stack) {
    if (isEmpty(stack)) {
        printf("Stack is empty!\n"); // 栈为空,无栈顶元素
        return -1;
    }
    return stack->data[stack->top]; // 返回栈顶元素,不修改栈的结构
}

// 打印栈中的元素(用于调试)
void printStack(Stack* stack) {
    if (isEmpty(stack)) {
        printf("Stack is empty!\n");
        return;
    }
    printf("Stack: ");
    for (int i = 0; i <= stack->top; i++) {
        printf("%d ", stack->data[i]);
    }
    printf("\n");
}

// 主函数用于测试栈的功能
int main() {
    Stack stack;
    initStack(&stack); // 初始化栈

    push(&stack, 10);
    push(&stack, 20);
    push(&stack, 30);

    printStack(&stack); // 打印栈中的元素

    printf("Top element: %d\n", peek(&stack)); // 获取栈顶元素

    while (!isEmpty(&stack)) {
        printf("Popped element: %d\n", pop(&stack)); // 依次出栈并输出元素
    }

    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2024年的十大技术趋势 - AI 等等

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

网优小工具-基站ID行列转换

网优小工具&#xff0d;基站ID行列转换 因在日常工作需要对基站ID批量行转列&#xff0c;以方便在网管上批量筛选指定网元&#xff0c;该小工具基于微软Power Query插件编写&#xff0c;工具方便、简洁、易用&#xff0c;共享出来以方便工作。 工作界面 &#xff11;.粘贴需筛…

学习VUE2第6天

一.请求拦截器 可以节流&#xff0c;防止多次点击请求 toast是单例 二.前置路由守卫 在Vue.js中&#xff0c;前置路由守卫是指在路由转换实际发生之前执行的钩子函数。这是Vue Router&#xff08;Vue.js官方的路由管理器&#xff09;提供的一种功能&#xff0c;允许开发者在用…

中兴UME网管LTE共享参数配置-PLMN添加

本文为中兴设备UME网管电联中频共享参数配置&#xff0c;PLMN添加参数配置部分&#xff0c;因UME与U&#xff13;&#xff11;网管添加PLMN配置区别较大&#xff0c;UME网管需同时配置运营商EN&#xff0d;DC策略&#xff0c;相关配置流程及参数配置如下文。 PLMN eNodeB CU …

与 Apollo 共创生态:观看7周年大会的心路历程

前言 在科技飞速发展的今天&#xff0c;自动驾驶技术已然成为行业创新的热点之一。作为一名长期关注自动驾驶领域的技术人员&#xff0c;我有幸见证了Apollo平台的成长与壮大。七年前&#xff0c;Apollo的诞生为我们带来了无尽的想象与期待&#xff1b;七年后的今天&#xff0…

【自研网关系列】过滤器链 -- 灰度发布过滤器

&#x1f308;Yu-Gateway&#xff1a;&#xff1a;基于 Netty 构建的自研 API 网关&#xff0c;采用 Java 原生实现&#xff0c;整合 Nacos 作为注册配置中心。其设计目标是为微服务架构提供高性能、可扩展的统一入口和基础设施&#xff0c;承载请求路由、安全控制、流量治理等…

【介绍下Unity编辑器扩展】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

ubuntu下安装配置python3.11

方案1 添加仓库&#xff1a; $ sudo add-apt-repository ppa:deadsnakes/ppa $ sudo apt update $ sudo apt install python3.11然后查看有多少个python版本已经安装了&#xff1a; ls -l /usr/bin/python*python2.7,python 3.8 ,python 3.11. 然后&#xff0c;设置系统默认…

Android4.4真机移植过程笔记(一)

1、RK源码编译 获取内核源码&#xff1a; git clone git172.28.1.172:rk3188_kernel -b xtc_ok1000 内核编译环境&#xff1a; 从172.28.1.132编译服务器的/data1/ZouZhiPing目录下拷贝toolchain.tar.gz&#xff08;交叉编译工具链&#xff09;并解压到与rk3188_kernel同级目…

Visual Studio中怎样更改Nuget程序包源

场景 Visual Studio 2019 在使用NuGet添加依赖包时&#xff0c;在预览中搜索不到程序包。 排查下NuGet的程序包源为本地。 将程序包源修改下。 实现 在解决方案上右击选择管理解决方案中的NuGet程序包(在 Visual Studio 中打开“工具”>“选项”>“NuGet 包管理器”…

【配置】Docker搭建JSON在线解析网站

云服务器打开端口8787 连接上docker运行 docker run -id --name jsonhero -p 8787:8787 -e SESSION_SECRETabc123 henryclw/jsonhero-webhttp://ip:8787访问 Github&#xff1a;地址

考研数学|《880题》不会做,怎么办?

如果880大部分都不会做&#xff0c;说明基础掌握太差&#xff0c;如果已经到了10月底&#xff0c;我建议直接刷知能行吧。因为我一个同学经历和你类似&#xff0c;最后通过一个月高强度刷知能行算是补救了一些。 对于刚开始准备考研的同学和上面的题主&#xff0c;我想聊几句建…

PCIe协议之RCB、MPS、MRRS详解

✨前言&#xff1a; PCIe总线的存储器写请求、存储器读完成等TLP中含有数据负载&#xff0c;即Data Payload。Data Payload的长度和MPS&#xff08;Max Payload Size&#xff09;、MRRS&#xff08;Max Read Request Size&#xff09;和RCB&#xff08;Read Completion Bounda…

物联网实战--平台篇之(二)基础搭建

目录 一、Qt工程创建 二、数据库知识 三、通信协议 四、名词定义 本项目的交流QQ群:701889554 物联网实战--入门篇https://blog.csdn.net/ypp240124016/category_12609773.html 物联网实战--驱动篇https://blog.csdn.net/ypp240124016/category_12631333.html 一、Qt工程…

请编写函数fun,该函数的功能是:统计各年龄段的人数。N个年龄通过调用随机函数获得,并放在主函数的age数组中;

本文收录于专栏:算法之翼 https://blog.csdn.net/weixin_52908342/category_10943144.html 订阅后本专栏全部文章可见。 本文含有题目的题干、解题思路、解题思路、解题代码、代码解析。本文分别包含C语言、C++、Java、Python四种语言的解法完整代码和详细的解析。 题干 请编…

考研管理类联考(专业代码199)数学基础【2】整式与分式

一、整式及其运算 1.常用乘法公式&#xff08;逆运算就是因式分解&#xff09; 公式扩展① 公式扩展② 公式扩展③ 2.整式除法定理 若整式 F(x) 除以x-a的余式为r(x)&#xff0c;则 F(x) (x -a) g(x) r(x) &#xff0c;故r(a)F(a)成立 二、指数和对数的运算性质 1.指数运算…

vue3、element-plus递归实现动态菜单

vue3、element-plus递归实现动态菜单 使用场景&#xff1a;动态菜单为什么使用递归递归在动态菜单中的实现 使用场景&#xff1a;动态菜单 动态菜单是指菜单项的数量和层次结构可能是动态的&#xff0c;通常来自后端或用户输入。这些菜单的特征包括&#xff1a; 多层嵌套&…

Mysql从入门到精通——Mysql知识点总结(基础篇)

参考视频 黑马程序员 MySQL数据库入门到精通i 题单推荐 入门 进阶 SQL语句类型 DDL:数据定义语言&#xff0c;用来定义数据库对象(数据库&#xff0c;表&#xff0c;字段)DML:数据操作语言&#xff0c;对数据库表中的数据进行增删改DQL:数据查询语言,用来查询数据库中表的…

Linux第二节--常见的指令介绍集合(持续更新中)

点赞关注不迷路&#xff01;&#xff0c;本节涉及初识Linux第二节&#xff0c;主要为常见的几条指令介绍。 Linux下基本指令 1. ls 指令 语法&#xff1a; ls [选项][目录或文件] 功能&#xff1a;对于目录&#xff0c;该命令列出该目录下的所有子目录与文件。对于文件&#…

VPot-Free一款功能强大的文字转语音工具 v2306

01 软件介绍 VPot-FREE是一款功能强大的免费文字转语音工具&#xff0c;为用户提供了便捷的语音合成功能。无论是想将文字转化为语音进行朗读&#xff0c;还是将文章转为语音保存&#xff0c;VPot-FREE都能满足你的需求。 多种语音风格可供选择&#xff0c;包括男声、女声以及…