C语言中的栈

news2025/2/28 1:12:38

一、栈的定义:

就是只能表的一端操作的顺序表或链表,允许操作的那一端成为栈顶元素,与之相对应的另一端称为栈底元素。

我们向栈里存入元素称为压栈,即最先放入的元素存放在栈底,最后放入的元素存放在栈顶。

我们将取出栈中的元素的操作称为出栈,即最先存入的元素最后取出,最后存入的元素最先取出。

二、对栈常规的操作

1.压栈(Push)

2.出栈(Pop)

3.获取栈顶元素(GetTop)

三、 栈的应用场景

1. 子程序的调用:在跳往子程序前,会先将下个指令的地址存到堆栈中,直到子程序执行完后再将地址取出,以回到原来的程序中。
2. 处理递归调用:和子程序的调用类似,只是除了储存下一个指令的地址外,也将参数、区域变量等数据存入堆栈中。
3. 处理中断事件:保存中断返回地址。
4. 表达式的转换[中缀表达式转后缀表达式]与求值(实际解决)。
5. 二叉树的遍历。
6. 图形的深度优先(depth一first)搜索法

四、顺序栈的实现

1.顺序栈的定义
#include<stdio.h>
#define MAXSIZE 10
typedef struct stack{
        int data[MAXSIZE];//数据域
        int top;//指向栈顶的位置的变量
}SqStack;
2.顺序栈的初始化
void InitSqStack(SqStack * S){
        S->top = -1;
}
3.获取栈顶元素
int GetTop(SqStack * S){
        //判空
        if(S->top == -1){
                printf("栈空");
                exit(0);
        }
        return S->data[S->top];
}
4.压栈
bool Push(SqStack * S,int x){
        //判满
        if(S->top == MAXSIZE){
                printf("栈满\n");
                return false;
        }
       S->top++;
       S->data[S->top] = x; 
       return true;
}
5.出栈
bool Push(SqStack * S,int * x){
       
        //判空
        if(S->top == -1){
                printf("栈空");
                return false;
        }

        *x = S->data[S->top];

        S->top--;

        return true;

}

五、链栈的实现

1.定义链栈的结构体:

typedef struct LinkNode{

    int data;

    struct LinkNode* next;

}LinkNode,*LinkStack;

2.链栈的初始化:

LinkStack InitLinkStack(LinkStack S){

    //为链栈的头结点开辟空间

    S = (LinkStack)malloc(sizeof(LinkNode));

    if(S == NULL){

        printf("初始化失败\n");

        return NULL;

    }

    S->data = 0;

    S->next = NULL;

    return S;

}

3.创建链栈:

栈是先进后出,链栈的创建使用链表的头插法实现

bool CreateLinkStack(LinkStack S){

    //判断链栈是否未初始化

    if(S == NULL){

        printf("链栈未初始化\n");

        return false;

    }

    //使用头插法创建链栈

    //创建一个新的节点

    int x;

    printf("请你输入插入的值\n");

    scanf("%d",&x);

    LinkNode* node = NULL;

    LinkNode* p = S;

    while(x != 999){

        node = (LinkNode*)malloc(sizeof(LinkNode));

        node->data = x;

        node->next = p->next;

        p->next = node;

        scanf("%d",&x);

    }

    return true;

}

4.链表的出栈:通过不断删除链表的第一个节点实现链表的出栈效果

bool PopLinkStack(LinkStack S,int *e){

    if(S->next == NULL){

        printf("该链栈为空栈\n");

        return false;

    }

    LinkNode* p = S;

    LinkNode* q = S->next;

    *e = q->data;

    p->next = q->next;

    free(q);

    q = NULL;

    return true;

}

5.获取链栈的栈顶元素:

bool GetElemStack(LinkStack S,int *e){

    if(S->next == NULL){

        printf("该链栈为空栈\n");

        return false;

    }

    *e = S->next->data;

    return true;

}

6.main函数:

int main(){

    LinkStack S;

    S = InitLinkStack(S);

    CreateLinkStack(S);

    int e;

    PopLinkStack(S,&e);

    printf("%d\t",e);

    PopLinkStack(S,&e);

    printf("%d\t",e);

    PopLinkStack(S,&e);

    printf("%d\t",e);

    GetElemStack(S,&e);

    printf("%d\n",e);

}

六、栈的应用

1.括号匹配:

使用顺序栈实现

主要思想:当输入左括号 [ ( { 就将其入栈,遇到 ] } ) 就将站内的元素出栈,若站为空直接匹配失败

若是中间遇到一个不匹配成功的就匹配失败,全部匹配成功才显示匹配成功。

#include<stdio.h>

#include<stdbool.h>

#include<string.h>

#define MAXSIZE 10

typedef struct Stack

{

    char data[MAXSIZE];

    int top;

}Stack;

/**1.初始化栈 */

bool InitStack(Stack * S){

    S->top = -1;

    return true;

}

/**2.压栈 */

bool PushStack(Stack * S,char x){

    if(S->top == MAXSIZE - 1){

        printf("栈满\n");

        return false;

    }

    S->top++;

    S->data[S->top] = x;

    return true;

}

/**3.出栈 */

bool PopStack(Stack * S,char * e){

    if(S->top == -1){

        printf("栈为空\n");

        return false;

    }

    *e = S->data[S->top];

    S->top--;

    return true;

}

/**4.括号匹配 */

bool CheckBracket(char * bracket){

    Stack S;

    InitStack(&S);

    PushStack(&S,'[');

    PushStack(&S,'(');

    int length = strlen(bracket);

    for(int i = 0; i < length; i++){

        if(bracket[i] == '(' || bracket[i] == '[' || bracket[i] == '{'){

            //扫描到左括号

            //压栈

            PushStack(&S,bracket[i]);

        }

        //扫描到右括号

        else if(bracket[i] == ')' || bracket[i] == ']' || bracket[i] == '}'){

            if(S.top == -1){

                //栈为空

                printf("栈为空,匹配失败\n");

                return false;

            }

            //出栈

            char ch;

            PopStack(&S,&ch);

            if(ch != '(' && bracket[i] == ')'){

                printf("匹配失败\n");

                return false;

            }

            if(ch != '[' && bracket[i] == ']'){

                printf("匹配失败\n");

                return false;

            }

            if(ch != '{' && bracket[i] == '}'){

                printf("匹配失败\n");

                return false;

            }  

        }

    }

    //匹配完当栈为空时表示匹配成功

    if(S.top == -1){

        printf("匹配成功\n");

        return true;

    }else{

        printf("匹配失败\n");

        return false;

    }

}

int main(){

    char bracket[] = ")]";

    CheckBracket(bracket);

}

运行结果:

为啥是匹配成功呢?:我传递的给括号匹配功能函数CheckBracket的字符为")]",而我的匹配函数CheckBracket中先压入"(["根据栈的出栈规律先出栈( 与 字符串中的 )匹配成功 再出栈 [ 与 字符串中的 ]匹配,匹配成功,全部匹配成功则输出匹配成功。

2.计算器的实现

顺序栈的实现:

#include<stdio.h>

#include<stdlib.h>

#include<stdbool.h>

#define MAXSIZE 20

typedef struct stack{

    int data[MAXSIZE];

    int top;

}SqStack;

/*1.初始化*/

bool InitSqStack(SqStack * S){

    S->top = -1;

    return true;

}

/**2.压栈 */

bool PushStack(SqStack * S,int x){

    if(S->top == MAXSIZE - 1){

        printf("栈满\n");

        return false;

    }

    S->top++;

    S->data[S->top] = x;

    return true;

}

/**3.出栈 */

int PopStack(SqStack * S){

    if(S->top == -1){

        printf("栈空\n");

        return 0;

    }

    int x = S->data[S->top];

    S->top--;

    return x;

}

/**4.判断运算符的优先级操作 */

int Priority(char u){

    switch (u)

    {

        case '+':

        case '-':

            return 1;

        case '*':

        case '/':

            return 2;

        default:

            return -1;

    }

}

/**5.运算操作 */

void PopPush(SqStack * Sdata,SqStack * Soper){

    //将Sdata数字栈中的两个数据出栈

    int x;

    int y;

    x = PopStack(Sdata);

    y = PopStack(Sdata);

    //将Soper运算符栈中的运算符出栈

    char u;

    u = PopStack(Soper);

    //计算两数之和

    int n;

    switch (u)

    {

    case '+':

        n = y+x;

        break;

    case '-':

        n = y-x;

        break;

    case '*':

        n = y*x;

        break;

    case '/':

        n = y/x;

        break;

    }

    PushStack(Sdata,n);

}

/**6.扫描输入的字符串,分情况不断的进行PopPush操作,得出最终结果 */

int count(char * str){

    //创建数据栈与运算符栈

    SqStack Sdata,Soper;

    //初始化

    InitSqStack(&Sdata);

    InitSqStack(&Soper);

    //对字符串进行遍历

    while(*str != '\0'){

        //遍历到的字符是数字

        if(*str >= '0' && *str <= '9'){

            int y = *str - '0';

            str++;

            //如果是多个数字呢

            while(*str >= '0' && *str <= '9'){

                y = y*10+(*str - '0');

                str++;

            }

            PushStack(&Sdata,y);

        }

        //如果栈为空,或字符为(,或字符的优先级大于栈顶字符的优先级就将其放入字符栈

        else if(Soper.top == -1 || *str == '(' || Priority(*str) > Priority(Soper.data[Soper.top])){

            PushStack(&Soper,*str);

            str++;

        }

        //如果字符为)并且字符栈的栈顶元素为( 就出将字符栈的栈顶元素出栈

        else if(*str == ')' && Soper.data[Soper.top] == '('){

            int ch;

            ch = PopStack(&Soper);

            str++;

        }

        //字符的优先级小于等于栈顶字符的优先级就将其出栈,并且数据栈的栈顶与其相邻的两个元素出栈计算

        else{

            PopPush(&Sdata,&Soper);

        }

       

    }

    //字符串表达式扫描完毕,再次使用PopPush清空数据栈与运算符栈并计算

    while(Soper.top > -1){

        PopPush(&Sdata,&Soper);

    }

    int n = Sdata.data[Sdata.top];

    return n;

}

int main(){

    char c[50];

    printf("请你输入表达式\n");

    scanf("%s",c);

    int n = count(c);

    printf("结果为:%d",n);

}

运行结果:

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

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

相关文章

怿星科技与您相约——2024 Testing Expo

汽车测试及质量监控博览会(中国)Testing Expo China-Automotive 怿星科技展位路线 届时欢迎莅临2057号展台&#xff01;

OpenCV图像滤波(16)应用分离式滤波器函数sepFilter2D()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 对图像应用分离式线性滤波器。 该函数对图像应用分离式线性滤波器。首先&#xff0c;src 的每一行都用 1D 内核 kernelX 进行滤波。然后&#x…

爬虫 Web Js 逆向:RPC 远程调用获取加密参数(1)WebSocket 协议介绍

RPC (Remote Procedure Call) 是远程调用的意思。 在 Js 逆向时&#xff0c;本地可以和浏览器以服务端和客户端的形式通过 WebSocket 协议进行 RPC 通信&#xff0c;这样可以直接调用浏览器中的一些函数方法&#xff0c;不必去在意函数具体的执行逻辑&#xff0c;可以省去大量…

计算机网络面试题汇总

文章目录 计算机网络基础计算机网络体系结构(网络分层模型)OSI 七层模型是什么?每一层的作用是什么?TCP/IP 四层模型是什么?每一层的作用是什么?五层体系结构以及对应的协议为什么网络要分层,分层的好处?常见网络协议有哪些,每一层常见协议有哪些?应用层有哪些常见的协…

【网编】——tcp编程

tcp流程 服务器 头文件&#xff1a; #include <sys/types.h> /* See NOTES */ #include <sys/socket.h> #include <errno.h> #include<stdio.h> #include <netinet/in.h> #include <netinet/ip.h> /* superset of previous */ #…

链接Mysql 报错connection errors; unblock with ‘mysqladmin flush-hosts‘错误的解决方法!亲测有效!

文章目录 前言一、使用 mysqladmin flush-hosts 命令解锁 IP 地址二、增加 max_connect_errors 参数三、检查连接错误的原因 前言 今天正常的对各大的测试服进行重启的时候发现每台服务器都启动失败&#xff01;查看日志发现每台服务器都报一下的错误 java.sql.SQLException:…

分布式光伏管理系统具有什么功能?有推荐吗?

1、项目进度管理 分布式光伏管理系统能够全面管理项目的进度&#xff0c;从初步沟通、收资踏勘、设计、施工到并网发电的全流程。系统通过可视化的项目进度管理工具&#xff0c;展示每一步的完成情况&#xff0c;包括员工跟进记录、关键节点时间等&#xff0c;帮助管理者从宏观…

windows下php安装kafka

下载zookeeper Kafka 依赖 Zookeeper 进行分布式协调&#xff0c;所以需要下载Zookeeper &#xff0c;当然你也可以使用kafka包里自带的一个默认配置的 Zookeeper。这里我们单独下载一个 访问Zookeeper官方下载页面在页面中找到最新的稳定版本&#xff0c;点击相应的下载链接…

Datawhale X 魔搭 AI夏令营第四期-魔搭生图task2学习笔记

精读代码 1.环境安装 !pip install simple-aesthetics-predictor!pip install -v -e data-juicer!pip uninstall pytorch-lightning -y !pip install peft lightning pandas torchvision!pip install -e DiffSynth-Studio 开始要先把代码中需要使用到的包、库用pip命令安装。…

Fomepay和Fomecard是不是跑路了?

最近几天&#xff0c;fomepay跑路的消息&#xff0c;很多人知道了&#xff0c;fomepay客服繁忙&#xff0c;也无法提现。更多的是&#xff0c;很多人刚充值&#xff0c;才看到跑路的消息&#xff0c;实在惨啊&#xff01; 而现在&#xff0c;如果之前没有登录过fomepay的话&…

安卓TV入门项目

android studio创建tv项目 下载android studio点此下载 配置环境变量&#xff1a; d盘新增Android文件夹&#xff0c;创建android-avd和android-sdk文件夹 环境变量名称&#xff1a;ANDROID_HOME 环境变量值&#xff1a;D:\Android\android-sdk 环境变量名称&#xff1a;ANDRO…

深入理解 Spring 三级缓存:解决单例 Bean 循环依赖的利器

目录 一、什么是循环依赖&#xff1f; 二、关于传说中的三级缓存 1.基本概念&#xff1a; 2.三级缓存是哪三级&#xff1f; 3.【举个例子】那三级缓存是怎么解决上述代码例子中的A、B互相依赖呢&#xff1f; 详细过程&#xff1a;&#xff08;理解用&#xff09; 简约版…

【专题】2024年7月人工智能AI行业报告合集汇总PDF分享(附原数据表)

原文链接:https://tecdat.cn/?p37350 随着人工智能技术的飞速发展&#xff0c;AI已经成为当今时代的重要驱动力。本报告将聚焦于人工智能AI行业的最新动态&#xff0c;涵盖客户服务、体验营销、资产管理以及国产AI大模型应用等多个领域。通过深入研究和分析&#xff0c;我们…

【Python学习-UI界面】PyQt5 小部件5-QCheckBox

样式如下: 当将QCheckBox对象添加到父窗口时&#xff0c;文本标签之前会出现一个矩形框。 和QRadioButton一样&#xff0c;它也是一个可选择的按钮。 它通常用于用户被要求选择一个或多个可用选项的场景。 不同于单选按钮&#xff0c;复选框默认情况下不是互斥的。 为了限制…

Golang 与 Java:编程语言比较及如何选择

Golang 与 Java&#xff1a;哪种语言更好&#xff1f;我们的详细比较指南涵盖了语法、性能和流行度方面的主要差异&#xff0c;以帮助您做出决定。 在规划项目时&#xff0c;有许多编程语言可供选择。但一开始就选择正确的语言是成功启动或交付的关键。选择错误的语言&#xff…

用R语言进行数据类型的检查和基础转换

下面内容摘录自《R 语言与数据科学的终极指南》专栏文章的部分内容&#xff0c;每篇文章都在 5000 字以上&#xff0c;质量平均分高达 94 分&#xff0c;看全文请点击下面链接&#xff1a; 4章8节&#xff1a;用R做数据重塑&#xff0c;行列命名和数据类型转换-CSDN博客 欢迎…

servlet的执行顺序

执行的时候Tomcat先初始化 然后调用 server 根据server来回调请求方式下面会追入源码解释 package com.haogu.servlet;import javax.servlet.ServletConfig; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.…

【C++二分查找】1954. 收集足够苹果的最小花园周长

本文涉及的基础知识点 C二分查找 LeetCode1954. 收集足够苹果的最小花园周长 给你一个用无限二维网格表示的花园&#xff0c;每一个 整数坐标处都有一棵苹果树。整数坐标 (i, j) 处的苹果树有 |i| |j| 个苹果。 你将会买下正中心坐标是 (0, 0) 的一块 正方形土地 &#xff…

Windows平台RTSP|RTMP播放器如何实现实时录像功能

技术背景 RTSP、RTMP直播播放&#xff0c;这里不再赘述&#xff0c;我们可以很轻松的实现毫秒级的延迟体验&#xff0c;这里讲的是如何实现RTSP、RTSP流的实时录像功能。 我们理解的录像&#xff0c;可能觉得&#xff0c;只要有个开始录像、停止录像接口就够了&#xff0c;实…

mysql速起架子

wget https://dev.mysql.com/get/Downloads/MySQL-8.0/mysql-8.0.21-linux-glibc2.12-x86_64.tar.xz 下载mysql tar xvJf mysql-8.0.21-linux-glibc2.12-x86_64.tar.xz 解压 mv mysql-8.0.21-linux-glibc2.12-x86_64 mysql-8.0 改名 去到bin目录 cd bin mkdir data gr…