数据结构和算法(2)---- Stack 的原理和实现

Stack 的定义和结构

栈(Stack)是仅限于在表尾进行插入和删除的线性表
我们把允许插入和删除的一端称为栈顶(top),另一端称为栈底(bottom),不含任何元素的栈称为空栈,栈也被称为先进后出(Last In First Out)的线性表,简称LIFO结构
栈的机构如下图所示:
stack_struct

栈的插入操作,称为进栈,也称为压栈,入栈。类似于子弹压入弹夹
栈的弹出操作,称为出栈,有的也叫做弹栈。类型于子弹从弹夹中弹出

Stack 的抽象数据类型
ADT Stack(队列)
Data:
    同线性表, 元素具有相同的类型,相邻的元素具有前驱和后继关系
Operation:
    InitStack(S*)
    DestroySatck(S*)
    isEmpty(S*)
    Peek(Q*)// 获取栈顶的元素值,但是不弹出栈
    Pop(Q*, *e)
    Push(Q*, e)
    StackSize(Q)
endADT
静态数组实现Stack
  • 建立一个 MAX_SIZE 的数组, 用于存放 Stack 中的元素
  • 建立int类型 stack->top 代表栈顶,当弹出元素时,取出stack->top位置的值,同时stack->top--指向下一个元素
  • 当压入元素时,stack->top++代表的index 存入新压入元素,并且stack->top指向新元素的位置
    staticArrayStack

参考实现代码:

#define MAX_SIZE 100

typedef struct {
    int data[MAX_SIZE];
    int top;
}Stack;

static void initStack(Stack* stack) {
    stack->top = -1;
}

static int isEmpty(Stack* stack) {
    return (stack->top == -1);
}

static int isFull(Stack* stack) {
    return (stack->top == MAX_SIZE -1);
}

static void push(Stack* stack, int value) {
    if(isFull(stack)) {
        fprintf(stderr, "stack is full. \n");
        return;
    }
    stack->top++;
    stack->data[stack->top] = value;
    //printf("stack top is %d.\n", stack->top);
}

static int pop(Stack* stack) {
    if(isEmpty(stack)) {
        printf("stack is empty. \n");
        return -1;
    }
    int item = stack->data[stack->top];
    stack->top--;
    return item;
}


static int peek(Stack* stack) {
    if(isEmpty(stack)) {
        fprintf(stderr, "stack is empty. \n");
        return -1;
    }
    return stack->data[stack->top];
}

static int modifyTop(Stack* stack, int value) {
    if(isEmpty(stack)) {
        fprintf(stderr, "stack is empty. \n");
    }
    stack->data[stack->top]= value;
}

static int stackSize(Stack* stack) {
    return stack->top+1;
}

int testbasicStackStaticArray(int agrc, char *argv[]) {
    {
        Stack teststack;
        initStack(&teststack);
        push(&teststack,100);
        push(&teststack,110);
        push(&teststack,120);

        printf("stack size is %d.\n",stackSize(&teststack));

        int value = pop(&teststack);
        printf("stack pop value is %d. size:%d\n",value,stackSize(&teststack));

        value =pop(&teststack);
        printf("stack pop value is %d. size:%d\n",value,stackSize(&teststack));

        value = pop(&teststack);
        printf("stack pop value is %d. size:%d\n",value,stackSize(&teststack));

        printf("stack pop value is %d.\n",pop(&teststack));

    }

    {
        Stack teststack;
        initStack(&teststack);
        push(&teststack,100);
        printf("stack size:%d peek value:%d \n", stackSize(&teststack), peek(&teststack));
        int value = pop(&teststack);
        printf("stack size:%d peek value:%d \n", stackSize(&teststack), peek(&teststack));
    }
}
单链表实现Stack
  • 建立一个单链表,包含指向栈顶的指针Stack->top
  • 当压入元素时,就是单链表的头部插入操作,先给新元素分配空间,然后将新元素的 next 指向 当前的Stack->top,最后更新Stack->top的值
  • 当弹出元素时,就是单链表的头部删除操作,首先释放当前Stack->top节点,然后将Stack->top更新到之前元素的下一个位置
    单链表实现栈
    参考代码实现如下:
struct node {
    int data;
    struct node *next;
};
typedef struct {
    struct node* top;
} Stack;

static int empty(Stack* stack) {
    return(stack->top == NULL);
}

static void initStack(Stack* stack){
    stack->top = NULL;
}

static  void push(Stack* stack, int item) {
    struct node *pnode;
    pnode = (struct node *)malloc(sizeof(struct node));
    if(pnode == NULL) {
        printf("malloc node failed!.\n");
        exit(1);
    }
    pnode->data = item;
    pnode->next = stack->top;
    stack->top = pnode;
}

static  int pop(Stack* stack) {
    int item;
    struct node *pnode;
    if(empty(stack)) {
        printf("stack is empty.\n");
        exit(1);
    } else {
        item = stack->top->data;
        pnode = stack->top;
        stack->top = stack->top->next;
        free(pnode);
    }
    return item;
}

static int stackSize(Stack* stack) {
    int count = 0;
    struct node *pnode = stack->top;
    if(empty(stack)) {
        return 0;
    } else {
        do {
            pnode = pnode->next;
            count++;
        }while(pnode != NULL);
    }
    return count;
}

int testbasicStackImplsingleLinkedList(int argc, char *argv[]) {
    {
        Stack teststack;
        initStack(&teststack);
        push(&teststack, 110);
        push(&teststack, 111);
        push(&teststack, 113);
        push(&teststack, 223);
        push(&teststack, 678);

        int stacksize = stackSize(&teststack);
        printf("stack size is %d.\n", stacksize);

        printf("stack pop value is %d.\n",pop(&teststack));
        printf("stack pop value is %d.\n",pop(&teststack));
        printf("stack pop value is %d.\n",pop(&teststack));

        // check failed
        printf("stack pop value is %d.\n",pop(&teststack));

        stacksize = stackSize(&teststack);
        printf("stack size is %d.\n", stacksize);

    }

    return 1;
}

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

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

相关文章

为什么能通过文本分析情感?

通过文本分析情感,通常称为情感分析(Sentiment Analysis)或意见挖掘(Opinion Mining),是自然语言处理(NLP)的一个分支。这项技术能够识别和提取文本中的主观信息(对呀&am…

Linux操作系统处理器调度基本准则和实现

1,基本概念 在多道程序系统中,进程的数量往往多于处理机的个数,进程争用处理机的情况就在所难免。处理机调度是对处理机进行分配,就是从就绪队列中,按照一定的算法(公平、低效)选择一个进程并将…

windows端口被占用问题,杀死进程

描述:端口被占用 在使用IntelliJ IDEA运行程序时,可能会遇到端口占用的情况,这通常由以下几个原因引起: 1、同一程序多次启动:如果你没有正确关闭之前运行的程序实例,再次尝试运行相同的程序时,…

数据库系统概论(超详解!!!) 第十四节 数据库恢复技术

1.事务的基本概念 1.事务 事务(Transaction)是用户定义的一个数据库操作序列,这些操作要么全做,要么全不做,是一个不可分割的工作单位。 事务和程序是两个概念, 在关系数据库中,一个事务可以是一条SQL语句&#xff…

Leetcode3185. 构成整天的下标对数目 II

Every day a Leetcode 题目来源&#xff1a;3185. 构成整天的下标对数目 II 解法1&#xff1a;哈希 本质思路类同经典的“两数之和”。枚举右&#xff0c;用哈希表维护左。 枚举 j&#xff0c;并维护 cnt[x] 表示所有满足 i < j 的下标 i 中&#xff0c;有几个 hours[i]…

5个视频人声分离方法:一键批量分离人声和背景音乐(操作指南)

视频人声分离指的是从视频文件中提取人声部分&#xff0c;将其与背景音乐分离。想要将视频人声分离&#xff0c;可以使用手机上的音频人声分离app、或电脑端专业的人声分离软件和在线剪辑工具实现&#xff0c;只需要导入文件就可以实现视频人声分离。 本文整理了以下几款视频人…

分治精炼宝库----归并排序应用( ´◔︎ ‸◔︎`)

目录 一.基本概念: 二.归并排序&#xff1a; 三.交易逆序对总数&#xff1a; 四.计算右侧小于当前元素的个数&#xff1a; 五.翻转对&#xff1a; 六.合并k个有序链表&#xff1a; 一.基本概念: &#x1f43b;在计算机科学中&#xff0c;分治法是一种很重要的算法。字面上的…

ImportError: No module named createrepo

我在用createrepo命令创建本地源时,出现如下: ImportError: No module named createrepo原因估计就是之前升级python2.6为2.7时导致(系统为centos7),看网上很多说, 修改/usr/share/createrepo/genpkgmetadata.py 第一行的python路径,但我试了根本无效 我是重新通过yu…

QtCreator/VS中制作带有界面的动态库

1、首先创建动态库项目 class UNTITLED25_EXPORT Untitled25 {public:Untitled25(); };2、直接右键创建同名窗口类进行覆盖 3、引入global头文件并添加到处宏</

Cadence 16.6与17.4个人学习版推荐

一. 简介与下载 Cadence个人学习版是基于Cadence官方发行的安装包做了适当精简和优化的二次打包版本&#xff0c;包括了Cpature原理图设计、PSpice 电路仿真以及Allegro PCB设计等以电子产品设计为主的主要功能&#xff0c;能满足绝大部分硬件工程师的使用需求。 学习版预先已…

SpringBoot使用AutoConfigure实现依赖库自动导入配置

我们知道导入配置有两种&#xff0c;一种是Value&#xff0c;一种是ConfigurationProperties&#xff0c;将对应的类标记为Component即可导入。但是被注解标识的类创建Bean有一个前提&#xff0c;只对启动类所在的包路径下的所有带有Component等注解的类才会创建Bean。如果我们…

高中生都知道:Mybatis-Plus如何生成内置Sql的?

一&#xff1a;文章背景 本文从源码的角度进行分析Mybatis-Plus&#xff0c;为了阅读且吸收的更顺利&#xff0c;希望读者有以下基础: 对Java、Spring、Mybatis、Mybatis-Plus有一定的了解/使用基础对底层源码有略微学习&#xff0c;哪怕是为了框架二开、框架扩展或知识储备本…

【C++】循环、控制流语句、指针

8、循环&#xff08;loops&#xff09;&#xff08;1&#xff09;for loops for循环非常灵活&#xff0c;可以做很多事情。上图红框框出来的代码块就是一个for循环。 for是关键字 for后面内容分为三部分&#xff0c;每部分用分号&#xff1b;隔开 第一部分A是变量的声明&…

20240623(26.0) 重要财经新闻

财经关注 ► 券商中国&#xff1a;北交所于6月21日晚间受理了3家企业的IPO申请。6月20日晚间&#xff0c;沪深交易所各受理了1家IPO申请。这也意味着&#xff0c;三大交易所IPO受理全部恢复。与此同时&#xff0c;三大交易所IPO上市委会议也已经全部重启。 ► 全球多地近期遭遇…

【数据挖掘】机器学习中相似性度量方法-切比雪夫距离

写在前面&#xff1a; 首先感谢兄弟们的订阅&#xff0c;让我有创作的动力&#xff0c;在创作过程我会尽最大能力&#xff0c;保证作品的质量&#xff0c;如果有问题&#xff0c;可以私信我&#xff0c;让我们携手共进&#xff0c;共创辉煌。 路虽远&#xff0c;行则将至&#…

群体优化算法---电磁共振优化算法(EROA)介绍包含示例滤波器设计

介绍 电磁共振优化算法&#xff08;Electromagnetic Resonance Optimization Algorithm, EROA&#xff09;是一种新型的元启发式优化算法&#xff0c;其灵感来源于电磁共振现象。电磁共振是一种物理现象&#xff0c;当一个系统在特定频率下响应最大时&#xff0c;这个频率被称…

数据结构和算法(1) ---- Queue 的原理和实现

Queue 的定义和结构 队列(Queue) 是只允许在一端进行插入&#xff0c;在另一端进行删除的线性表 队列是一种先进先出(First In First Out)的线性表&#xff0c;简称 FIFO(First IN First OUT), 允许插入的一端称为队尾, 允许删除的一端称为队列头 队列的基本结构如下图所示&a…

使用python绘制三维曲面图

使用python绘制三维曲面图 三维曲面图三维曲面图的用途效果代码 三维曲面图 三维曲面图是一种用于展示三维数据的图表&#xff0c;通过一个连续的曲面来表示数据的变化情况。它通常用于可视化数学函数或实验数据的三维关系&#xff0c;可以非常直观地展示变量之间的复杂关系。…

大电流与小电流在检测原理上有区别吗

1 常用电流检测原理 1.1 分流器原理 被测量的电流在输入端电阻上Rshunt形成电压正比于测量电流&#xff0c;通过同相比例电路进行放大输出。 缺点&#xff1a; 输入电流减小时&#xff0c;需要更大的Rshunt&#xff1b;输入电阻Rshunt串入检测回路内将引起被测电流减小&a…

App推广告别邀请码,Xinstall助您一键触达海量用户!

在移动互联网高速发展的今天&#xff0c;App的推广与运营已成为每个开发者都必须面对的问题。然而&#xff0c;随着互联网流量的日益分散和用户需求的不断变化&#xff0c;传统的App推广方式已经难以满足现代市场的需求。尤其是在获取用户时&#xff0c;很多开发者还在采用传统…