数据结构数组栈的实现

news2025/1/23 11:51:49

在这里插入图片描述

Hello,今天我们来实现一下数组栈,学完这个我们又更进一步了。

一、栈

栈的概念

栈是一种特殊的线性表,它只允许在固定的一端进行插入和删除元素的操作。
进行数据的插入和删除只在栈顶实现,另一端就是栈底。
栈的元素是后进先出。

压栈:栈的数据进入就是压栈
出栈:栈的数据删除就叫出栈

我们画一个原理图让大家比较好理解一下。

在这里插入图片描述

这一过程叫做pop出栈

我们上述的过程都是在栈顶实现出栈入栈,并不能像顺序表和单链表那样从任意位置删除和增加,但这就是栈的性质,我们后面会讲它的作用。

实现栈我们可以用链表产生节点的方式链接他们,但是也可以用数组下标访问的方式,类似顺序表这样的方法
那这两个方法哪个好呢,我们来比较一下。

因为栈的性质我们不得从栈顶出栈和入栈,如果我们实现的时候是链表的方式,那必然会存在一个问题,就是我们的时间复杂度是O(N)我们需要遍历一遍数组,这样的话栈好像变得“土”,所以用数组的方式更快的提高效率,

二、栈的定义

typedef int StackDataType;
#define N 100
struct Stack
{
	StackDataType arry[N];
	StackDataType top;
};

这是静态栈,在顺序表的时候我们就讲过静态栈存在缺点,最大的缺点就是不能开辟空间,100个最多只能存100个数据,如果我只使用10个int空间就存在浪费了,如果我要存储1000个数据,我们的空间又不够了,这就会造成一系列问题,所以我们改一下,变成动态栈,我们来实现一下吧。

typedef int StackDataType;
typedef struct Stack
{
	StackDataType* arry;
	int top;
	int capacity;
}Stack;

有了结构体还是老样子,我们来实现一下接口函数,开整!
初始化栈

void StackInit(Stack* pst)
{
	assert(pst);
	pst->arry = NULL;
	pst->capacity = pst->top = 0;
}

初始化栈这个大家肯定会了。

销毁


void StackDestory(Stack* pst)
{
	assert(pst);
	free(pst->arry);
	pst->capacity = pst->top = 0;
	
}

判断栈是否为空

bool StackEmpty(Stack* pst)
{
	assert(pst);
	return pst->top == 0;
}

现在我们要实现一个入栈的方法,入栈的时候我们需要检查一下我们的内存空间是不是满了,和顺序表一样的道理,如果满了我们就需要扩容。所以在入栈的时候需要判断一下它空间有没有满。

void StackPush(Stack* pst, StackDataType x)
{
	assert(pst);
	if (pst->capacity == pst->top)
	{
		int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;
		StackDataType* tmp = (StackDataType*)realloc(pst->arry, sizeof(int) * newcapacity);
		if (tmp == NULL)
		{
			printf("realloc fail\n");
			exit(-1);
		}
		pst->arry = tmp;
		pst->capacity = newcapacity;
	}
	pst->arry[pst->top - 1] = x;
	pst->top++;

}

这和我们顺序表的尾插一摸一样,现在看大家肯定觉得简单很多了。
有了入栈,那就有出栈。

void StackPop(Stack* pst)
{
	assert(pst);
	if (pst->top > 0)
	{
		pst->top--;
	}
}

因为我们上面写了一个判断该数是不是为空我们也可以写成

void StackPop(Stack* pst)
{
	assert(pst);
	if (!StackEmpty(pst))
	{
		pst->top--;
	}
}

返回栈顶数据

StackDataType StackTop(Stack* pst)
{
	assert(pst);
	return pst->arry[pst->top - 1];
}

统计栈里有多少数

int StackSize(Stack* pst)
{
	assert(pst);
	return pst->top;
}

完整代码

#pragma once

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<stdbool.h>
//typedef int StackDataType;
//#define N 100
//struct Stack
//{
//	StackDataType arry[N];
//	StackDataType top;
//};


typedef int StackDataType;
typedef struct Stack
{
	StackDataType* arry;
	int top;
	int capacity;
}Stack;

void StackInit(Stack* pst);

void StackDestory(Stack* pst);

bool StackEmpty(Stack* pst);

void StackPush(Stack* pst, StackDataType x);

void StackPop(Stack* pst);

StackDataType StackTop(Stack* pst);

int StackSize(Stack* pst);

#include"Stack.h"


void StackInit(Stack* pst)
{
	assert(pst);
	pst->arry = NULL;
	pst->capacity = pst->top = 0;
}

void StackDestory(Stack* pst)
{
	assert(pst);
	free(pst->arry);
	pst->capacity = pst->top = 0;
	
}

bool StackEmpty(Stack* pst)
{
	assert(pst);
	return pst->top == 0;
}

void StackPush(Stack* pst, StackDataType x)
{
	assert(pst);
	if (pst->capacity == pst->top)
	{
		int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;
		StackDataType* tmp = (StackDataType*)realloc(pst->arry, sizeof(int) * newcapacity);
		if (tmp == NULL)
		{
			printf("realloc fail\n");
			exit(-1);
		}
		pst->arry = tmp;
		pst->capacity = newcapacity;
	}
	pst->arry[pst->top - 1] = x;
	pst->top++;

}

void StackPop(Stack* pst)
{
	assert(pst);
	if (pst->top > 0)
	{
		pst->top--;
	}
}

void StackPop(Stack* pst)
{
	assert(pst);
	if (!StackEmpty(pst))
	{
		pst->top--;
	}
}

StackDataType StackTop(Stack* pst)
{
	assert(pst);
	return pst->arry[pst->top - 1];
}

int StackSize(Stack* pst)
{
	assert(pst);
	return pst->top;
}

栈的应用也有很多,后面会分享给大家,我们下次再见

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

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

相关文章

vue2 生命周期,工程化开发入门

一、今日目标 1.生命周期 生命周期介绍生命周期的四个阶段生命周期钩子声明周期案例 2.工程化开发入门 工程化开发和脚手架项目运行流程组件化组件注册 二、Vue生命周期 思考&#xff1a;什么时候可以发送初始化渲染请求&#xff1f;&#xff08;越早越好&#xff09;什么…

【线程池】ThreadPoolExecutor的使用示例

文章目录 通过ThreadPoolExecutor创建线程池。线程的处理结果如何获取&#xff1f; 通过ThreadPoolExecutor创建线程池。 ThreadPoolExecutor构造方法参数&#xff1a; int corePoolSize //核心线程数量int maximumPoolSize//最大线程数long keepAliveTime//当线程数大于核心…

【谷粒学院】开发篇二:后台管理系统搭建逆向生成代码

后台管理系统介绍 本篇文章主要内容如下&#xff1a; 1.使用人人开源人人开源绞手架搭建后台管理系统的前端和后端框架。 2.使用renren-generator逆向生成微服务&#xff08;gulimall_pms、gulimall_oms、gulimall_sms、gulimall_ums、gulimall_wms&#xff09;的CRUD代码。 …

前端高频面试题 js中堆和栈的区别和浏览器的垃圾回收机制

一、 栈(stack)和 堆(heap) 栈(stack)&#xff1a;是栈内存的简称&#xff0c;栈是自动分配相对固定大小的内存空间&#xff0c;并由系统自动释放&#xff0c;栈数据结构遵循FILO&#xff08;first in last out&#xff09;先进后出的原则&#xff0c;较为经典的就是乒乓球盒结…

机器学习---LDA代码

1. 获取投影坐标 import numpy as npdef GetProjectivePoint_2D(point, line):a point[0]b point[1]k line[0]t line[1]if k 0: return [a, t]elif k np.inf: return [0, b]x (ak*b-k*t) / (k*k1)y k*x treturn [x, y] 该函数用于获取一个点到一条直线的投影点…

以物联网为核心的智慧工地云平台:聚集智能技术,实现建筑工地智慧管理

智慧工地云平台源码&#xff0c;智慧工地项目监管平台源码&#xff0c;智慧工地可视化数据大屏源码 智慧工地云平台是将云计算、大数据、物联网、移动技术和智能设备等信息化技术手段&#xff0c;聚集在建筑工地施工管理现场&#xff0c;围绕人员、机械、物料、环境等关键要素&…

网络安全(黑客)——自学日薪2700

以下是练习舞蹈时长两年半的苕皮哥的故事 你想想一个跨专业的都能拿到日薪2700&#xff0c;你上你也行&#xff0c;那么接下来就是我给大家&#xff0c;整理的网络安全学习思路&#xff0c;让大家斩获高薪&#xff01; 前言&#xff1a; 想自学网络安全&#xff08;黑客技术&am…

秦丝九周年 | 秦丝生产ERP,全新一代服装ERP、SCM

秦丝九周年的主题是“数字生意&#xff0c;经九长兴”&#xff0c;通过数字化工具帮助更多商家开启数字生意&#xff0c;改变传统的低效率的工作&#xff0c;提供更加简单高效地把生意做大做强。 在帮助200万商家通过数字化工具管理零售、批发、连锁门店的同时&#xff0c;秦丝…

_数字矩阵

题目&#xff1a;一个3阶的数字矩阵如下&#xff1a; 1 2 3 8 9 4 7 6 5 现在给定数字n(1<n≤20)&#xff0c;输出n阶数字矩阵。 思路&#xff1a; 放出一条好玩的贪吃蛇&#xff0c;按照右下左上的顺序吃蛋糕&#xff0c;一边吃蛋糕&#xff0c;一边拉数字&#xff1b…

[C++ 网络协议] 套接字的多种可选项

目录 1. 套接字的可选项 2. 获取/设置套接字可选项 2.1 getsockopt函数&#xff08;获取套接字可选项&#xff09; 2.2 setsockopt函数&#xff08;设置套接字可选项&#xff09; 3. 常用套接字可选项 3.1 SOL_SOCKET协议层的SO_TYPE可选项 3.2 SOL_SOCKET协议层的SO_SN…

【八股】2023秋招八股复习笔记4(MySQL Redis等)

文章目录 目录1、MySQLmysql索引实现mysql索引优化mysql索引失效的情况mysql 千万数据优化mysql 事务隔离级别 & 实现原理mysql MVCC版本链&#xff08;undo log&#xff09;mysql数据同步机制 & 主从复制 &#xff08;binlog&#xff09;mysql 日志&数据恢复&…

git及GitHub的使用

文章目录 git在本地仓库的使用github使用创建仓库https协议连接(不推荐&#xff0c;现在用起来比较麻烦)ssh连接&#xff08;推荐&#xff09;git分支操作冲突处理忽略文件 git在本地仓库的使用 1.在目标目录下右键打开git bash here 2.创建用户名和邮箱(注&#xff1a; 下载完…

户外跑步用什么耳机、户外运动耳机推荐

跑步是一项简单的运动&#xff0c;只需要交替迈左右腿就可以进行。然而&#xff0c;跑步有时可能变得单调乏味。即使是意志坚定、热爱跑步的人&#xff0c;在这个漫长的过程中也会感到乏味&#xff0c;更不用说像你我这样的普通跑者了。音乐能够让跑步变得更加有趣&#xff0c;…

Java小白基础自学阶段(持续更新...)

引言 Java作为一门广泛应用于企业级开发的编程语言&#xff0c;对初学者来说可能会感到有些复杂。然而&#xff0c;通过适当的学习方法和资源&#xff0c;即使是小白也可以轻松掌握Java的基础知识。本文将提供一些有用的建议和资源&#xff0c;帮助小白自学Java基础。 学习步骤…

【80天学习完《深入理解计算机系统》】第十天 3.3 条件码寄存器【CF ZF SF OF】【set】

专注 效率 记忆 预习 笔记 复习 做题 欢迎观看我的博客&#xff0c;如有问题交流&#xff0c;欢迎评论区留言&#xff0c;一定尽快回复&#xff01;&#xff08;大家可以去看我的专栏&#xff0c;是所有文章的目录&#xff09;   文章字体风格&#xff1a; 红色文字表示&#…

R语言之数据框的合并

文章和代码已经归档至【Github仓库&#xff1a;https://github.com/timerring/dive-into-AI 】或者公众号【AIShareLab】回复 R语言 也可获取。 文章目录 1.纵向合并&#xff1a;rbind( )2. 横向合并&#xff1a;cbind ( )3. 按照某个共有变量合并&#xff1a;merge( )full_joi…

测开面经分享(偏Python)

某基金管理公司线下测试开发面试题总结。 预计阅读时间&#xff1a; 25分钟 测开题目如下 可以尝试自己先写&#xff0c;写完之后再去看参考解法哦 ~ 1、编写一段代码&#xff0c;把 list 的数平方(语言不限) ListA [1, 3, 5, 7, 9, 11] 2、使用 Python 语言编写一个日志…

如何把本地项目上传github

一、在gitHub上创建新项目 【1】点击添加&#xff08;&#xff09;-->New repository 【2】填写新项目的配置项 Repository name&#xff1a;项目名称 Description &#xff1a;项目的描述 Choose a license&#xff1a;license 【3】点击确定&#xff0c;项目已在githu…

大语言模型之五 谷歌Gemini

近十年来谷歌引领着人工智能方向的发展&#xff0c;从TensorFlow到TPU再到Transformer&#xff0c;都是谷歌在引领着&#xff0c;然而&#xff0c;在大语言模型上&#xff0c;却被ChatGPT&#xff08;OpenAI&#xff09;抢了风头&#xff0c;并且知道GPT-4&#xff08;OpenAI&a…

基于Python的网上宠物用品销售网站SpringBoot+Vue宠物用品商城系统源码+lw

&#x1f495;&#x1f495;作者&#xff1a;计算机源码社 &#x1f495;&#x1f495;个人简介&#xff1a;本人七年开发经验&#xff0c;擅长Java、Python、PHP、.NET、微信小程序、爬虫、大数据等&#xff0c;大家有这一块的问题可以一起交流&#xff01; &#x1f495;&…