C语言数据结构与算法之栈练习2:判断栈是否为空或者是否已满

2024-05-11 16:12

本文主要是介绍C语言数据结构与算法之栈练习2:判断栈是否为空或者是否已满,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

练习2:判断栈是否为空或者是否已满

需求1:实现一个简单的栈结构,并实现入栈和出栈的功能,编写相关的代码进行测试。

需求2:如果栈已满,则不要再继续入栈,并提示错误信息。如果栈为空,则不要出栈,并提示错误信息。错误信息要输出到标准错误流。

需求3:编写代码,分别覆盖栈已满和栈为空的情况。

完整代码
#include <stdio.h>
#include <stdlib.h>typedef struct stack {int* arr;int cap;int top;
} Stack;Stack* newStack(int cap){Stack* stack = malloc(sizeof(Stack));stack -> arr = malloc(sizeof(int) * cap);stack -> cap = cap;stack -> top = 0;return stack;
}void freeStack(Stack* stack){free(stack -> arr);stack -> cap = 0;stack -> top = 0;free(stack);stack = NULL;
}int isFullStack(Stack* stack){return stack != NULL && stack -> top == stack -> cap;
}int isEmptyStack(Stack* stack){return stack != NULL && stack -> top == 0;
}void pushStack(Stack* stack, int value){if (isFullStack(stack)){fprintf(stderr, "the stack is full\n");return;}stack -> arr[stack -> top] = value;stack -> top ++;
}int popStack(Stack* stack){if (isEmptyStack(stack)){fprintf(stderr, "the stack is empty\n");return -1;}int value = stack -> arr[stack -> top - 1];stack -> top --;return value;
}int main(){Stack* stack = newStack(3);pushStack(stack, 111);pushStack(stack, 222);pushStack(stack, 333);pushStack(stack, 333333);int value;value = popStack(stack);printf("value=%d\n", value);value = popStack(stack);printf("value=%d\n", value);value = popStack(stack);printf("value=%d\n", value);value = popStack(stack);printf("value=%d\n", value);freeStack(stack);return 0;
}

输出结果:

the stack is full
value=333
value=222
value=111
the stack is empty
value=-1
核心代码:栈结构

定义栈类型的数据结构:typedef struct stack {

声明一个int类型的地址,用来存储栈的内容:int* arr;

声明栈的容量:int cap;

声明栈顶:int top;

给栈类型的结构体取个别名:} Stack;

typedef struct stack {int* arr;int cap;int top;
} Stack;
核心代码:新建栈方法

定义新建栈的方法:Stack* newStack(int cap){

从内存中开辟一个空间,用来存储栈的信息:Stack* stack = malloc(sizeof(Stack));

从内存中开辟一个空间,用来存储栈数组容器的信息:stack -> arr = malloc(sizeof(int) * cap);

记录栈的容量:stack -> cap = cap;

记录栈顶:stack -> top = 0;

返回新建的栈:return stack;

Stack* newStack(int cap){Stack* stack = malloc(sizeof(Stack));stack -> arr = malloc(sizeof(int) * cap);stack -> cap = cap;stack -> top = 0;return stack;
}
核心代码:释放栈方法

声明释放栈的方法:void freeStack(Stack* stack){

释放栈的数组容器内存:free(stack -> arr);

栈的容量归零:stack -> cap = 0;

栈顶归零:stack -> top = 0;

释放栈本身的内存:free(stack);

将栈的地址置空,防止野指针:stack = NULL;

void freeStack(Stack* stack){free(stack -> arr);stack -> cap = 0;stack -> top = 0;free(stack);stack = NULL;
}
核心代码:判断栈是否已满方法

如果栈不是空指针,并且栈顶位置和容器容量相同,则说明栈已经满了。

正常情况下,栈顶位置应该小于栈容量。

int isFullStack(Stack* stack){return stack != NULL && stack -> top == stack -> cap;
}
核心代码:判断栈是否为空方法

如果栈不是空指针,或者栈顶位置在0,则说明这个栈是个空栈。

int isEmptyStack(Stack* stack){return stack != NULL && stack -> top == 0;
}
核心代码:入栈方法

声明入栈的方法:void pushStack(Stack* stack, int value){

判断栈是否已满:if (isFullStack(stack)){

将数据添加到栈顶的位置:stack -> arr[stack -> top] = value;

栈顶的位置自增:stack -> top ++;

void pushStack(Stack* stack, int value){if (isFullStack(stack)){fprintf(stderr, "the stack is full\n");return;}stack -> arr[stack -> top] = value;stack -> top ++;
}
核心代码:出栈方法

声明出栈的方法:int popStack(Stack* stack){

判断栈是否为空:if (isEmptyStack(stack)){

获取栈顶的元素:int value = stack -> arr[stack -> top - 1];

栈顶自减:stack -> top --;

返回获取到的元素:return value;

int popStack(Stack* stack){if (isEmptyStack(stack)){fprintf(stderr, "the stack is empty\n");return -1;}int value = stack -> arr[stack -> top - 1];stack -> top --;return value;
}
核心代码:测试方法

新建容量为3的栈对象:Stack* stack = newStack(3);

向栈中追加四条数据:pushStack(stack, 111);

弹出四次栈中的栈顶元素:int value = popStack(stack);

查看栈顶元素:printf("value=%d\n", value);

释放栈:freeStack(stack);

int main(){Stack* stack = newStack(3);pushStack(stack, 111);pushStack(stack, 222);pushStack(stack, 333);pushStack(stack, 333333);int value;value = popStack(stack);printf("value=%d\n", value);value = popStack(stack);printf("value=%d\n", value);value = popStack(stack);printf("value=%d\n", value);value = popStack(stack);printf("value=%d\n", value);freeStack(stack);return 0;
}

这篇关于C语言数据结构与算法之栈练习2:判断栈是否为空或者是否已满的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/980023

相关文章

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

如何通过try-catch判断数据库唯一键字段是否重复

《如何通过try-catch判断数据库唯一键字段是否重复》在MyBatis+MySQL中,通过try-catch捕获唯一约束异常可避免重复数据查询,优点是减少数据库交互、提升并发安全,缺点是异常处理开... 目录1、原理2、怎么理解“异常走的是数据库错误路径,开销比普通逻辑分支稍高”?1. 普通逻辑分支 v

GO语言短变量声明的实现示例

《GO语言短变量声明的实现示例》在Go语言中,短变量声明是一种简洁的变量声明方式,使用:=运算符,可以自动推断变量类型,下面就来具体介绍一下如何使用,感兴趣的可以了解一下... 目录基本语法功能特点与var的区别适用场景注意事项基本语法variableName := value功能特点1、自动类型推

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

从基础到进阶详解Python条件判断的实用指南

《从基础到进阶详解Python条件判断的实用指南》本文将通过15个实战案例,带你大家掌握条件判断的核心技巧,并从基础语法到高级应用一网打尽,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录​引言:条件判断为何如此重要一、基础语法:三行代码构建决策系统二、多条件分支:elif的魔法三、

Linux实现查看某一端口是否开放

《Linux实现查看某一端口是否开放》文章介绍了三种检查端口6379是否开放的方法:通过lsof查看进程占用,用netstat区分TCP/UDP监听状态,以及用telnet测试远程连接可达性... 目录1、使用lsof 命令来查看端口是否开放2、使用netstat 命令来查看端口是否开放3、使用telnet

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

redis数据结构之String详解

《redis数据结构之String详解》Redis以String为基础类型,因C字符串效率低、非二进制安全等问题,采用SDS动态字符串实现高效存储,通过RedisObject封装,支持多种编码方式(如... 目录一、为什么Redis选String作为基础类型?二、SDS底层数据结构三、RedisObject

Go语言使用Gin处理路由参数和查询参数

《Go语言使用Gin处理路由参数和查询参数》在WebAPI开发中,处理路由参数(PathParameter)和查询参数(QueryParameter)是非常常见的需求,下面我们就来看看Go语言... 目录一、路由参数 vs 查询参数二、Gin 获取路由参数和查询参数三、示例代码四、运行与测试1. 测试编程路

Go语言使用net/http构建一个RESTful API的示例代码

《Go语言使用net/http构建一个RESTfulAPI的示例代码》Go的标准库net/http提供了构建Web服务所需的强大功能,虽然众多第三方框架(如Gin、Echo)已经封装了很多功能,但... 目录引言一、什么是 RESTful API?二、实战目标:用户信息管理 API三、代码实现1. 用户数据