首页 > 其他分享 >数据结构之——栈

数据结构之——栈

时间:2023-05-27 13:02:03浏览次数:58  
标签:ps top 栈顶 ST assert STDataType 数据结构

@TOC


前言

本文主要讲述特殊的线性表——栈:


栈是什么,栈的特点

数据结构中有一种特殊的线性表叫栈。 栈有一种特点:

它允许后进入的数据先拿出来。

类似一个弹夹,或是一个装砖头的容器。

数据结构之——栈_初始化

栈的基本操作有如上图:

类比一个缸,缸的底端是栈底,顶端是栈顶,放入数据称为入栈,取出数据称为出栈。

对于一个栈,其特点为后进先出Last In Fist Out 或者 先进后出Fist Out Last In

所以在实现栈这种特殊的线性表时,当我们拿出数据的时候,只需要取栈顶元素即可,存入数据时,只需往栈顶存放数据。

综合栈的特点,在实现其结构时更适合使用数组 ,而不是链表。 当然使用链表也是可行的,相比而言: 使用链表的缺点有: 1.在压栈时总需要next的指针来维护。2。出栈时需要记录上一个节点的位置,效率较低。

数组的方式可以解决上述两个问题,且无其他较为严重的缺点。

下面来实现栈的基本功能:

实现栈的基本操作

栈的相关操作声明

void StackInit(ST* ps);//初始化
void StackDestroy(ST* ps);
void CheckCapacity(ST** ps);//检查容量
void StackPush(ST* ps,STDataType x);//插入元素
void StackPop(ST* ps);//删除栈顶元素
int StackSize(ST* ps);//计算栈有多少个数据
bool StackEmpty(ST* ps);//判断栈是否为空
STDataType StackTop(ST* ps);//取栈顶元素

注:ps是指向一个结构体的指针,在main函数创建了一个结构体:
	ST st;
	并且传参传的是结构体的地址

1.创建栈

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>

typedef int STDataType;
//用顺序表实现栈

typedef struct Stack
{
	STDataType* a;
	int top;//插入栈顶元素
	int capacity;//栈的容量
}ST;

解读: 定义一个结构体:使用结构体的指针来维护数组栈top表示栈顶的元素 capacity表示栈的容量大小

数据结构之——栈_初始化_02

2.对栈进行初始化

void StackInit(ST* ps)//初始化
{
	assert(ps!=NULL);
	ps->a = NULL;
	ps->top = ps->capacity = 0;
	//ps->top可以初始化成-1,此时先++,再赋值
	//此时指向的就是栈顶元素
}

3.销毁栈

void StackDestroy(ST* ps)
{
	assert(ps);
	free(ps->a);
	ps->a = NULL;
	ps->top = ps->capacity = 0;
}

4.判断栈是否为空

bool StackEmpty(ST* ps)//判断栈是否为空
{
	assert(ps);
	return ps->top == 0;
}

5.压栈操作

void CheckCapacity(ST**ps)//检查容量
{
	assert(ps != NULL);
	if ((*ps)->top == (*ps)->capacity)
	{
		STDataType newcapacity = (*ps)->capacity == 0 ? 4 : (*ps)->capacity * 2;
		STDataType* tmp = (STDataType*)realloc((*ps)->a,(sizeof(STDataType)*newcapacity));//申请的空间是存放STDataType的
		//不是用来存放结构体的
		//如果第一个参数是一个NULL,realloc的作用就跟malloc一样,所以可以传NULL
		assert(tmp != NULL);
		(*ps)->a = tmp;// 把新地址给ps->a
		(*ps)->capacity = newcapacity;
	}
}

void StackPush(ST* ps, STDataType x)//插入元素
{
	assert(ps);
	CheckCapacity(&ps);//这里如果传参传的是ps,相当于传值调用,在CheckCapacity函数内部申请的空间就无法返回来了。
	ps->a[ps->top] = x; // 先赋值,再++,因为ps->top初始化是0,就是指向栈顶元素的下一个。
	ps->top++;
}

解读:入栈时首先需要检查栈空间的容量大小,如果栈空间不足则需增容。在这里分装了一个检查容量的函数。

6.删除栈顶元素

void StackPop(ST* ps)//删除栈顶数据
{
	assert(ps);
	assert(!StackEmpty(ps));

	ps->top--;
}

7.取出栈顶元素

注:第六个函数仅仅删除栈顶元素并未拿到该数据这里分装函数是更方便的

STDataType StackTop(ST* ps)//取栈顶元素
{
	assert(ps);
	assert(!StackEmpty(ps)); //感叹号表达式让语句的逻辑相反

	return ps->a[ps->top - 1];
	//在这里我初始化top为0,表示指向栈顶元素的下一个位置
	//因为初始化为0时,需要先压栈,top值再++,此时就指向了下一个位置了
}

8.计算栈内存放多少个数据

int StackSize(ST* ps)//计算栈有多少个数据
{
	assert(ps);
	assert(!StackEmpty(ps));
	return ps->top;
}

解读:这里直接返回top,而不是返回top-1 因为我们在初始化的时候是将top置为0,先入栈,top再++ 假如入栈三次,那么top此时就是3,但是top指向的位置是入栈的第三个元素的下一个位置。

函数基本功能如下:

数据结构之——栈_ci_03

总结

掌握了链表的结构之后,实现栈难度是不大的。

标签:ps,top,栈顶,ST,assert,STDataType,数据结构
From: https://blog.51cto.com/u_15818575/6361993

相关文章

  • 用Python开发输入法后台(5)——数据结构
    全部汉字我从网上收集了一些资料,构建了一个<全部汉字.json>文件,文件格式如下所示:{"吖":[["aa","ya"],"szhdps"],"呵":[["aa",......
  • 数据结构与算法—排序算法篇
    1、选择排序1.1、算法思想每趟从待排序的数据元素中,选出最小(或最大)的一个元素,顺序放在待排序的数列最前面,直到全部待排序的数据元素排完。1.2、步骤1、查找此数列中的最小元素,将其与第一个交换2、从第二个值开始找,查找到最小元素,将其与第二个交换3、以此类推,直到遍历结束1.3、算法......
  • 基础数据结构方法汇总
    字符串方法:mystr.capitalize()第一个字符转换为大写,其它都转为小写(本来的大写字母也转为小写)"abCd"-->Abcd 列表方法:lst.count(obj)lst.append(obj)lst.extend(obj)lst.index(obj)元素obj不存在,则会引发ValueError异常lst.insert(下标,obj)如果下标不存在,则会插......
  • kissat分析01_基本数据结构03_frame_trail
      frame.h1#defineINVALID_TRAILUINT_MAX23structframe4{5unsigneddecision;6unsignedtrail:LD_MAX_TRAIL;7unsignedused:2;8boolpromote:1;9};1011//*INDENT-OFF*1213typedefSTACK(frame)frames;1415//*I......
  • 【前端算法学习】数据结构之“栈”
    JS中最棒的数据结构:数组​ 数组是计算机科学中最常用的数据结构。我们知道,可以在数组的任意位置上删除或添加元素。然而,有时候我们还需要一种在添加或删除元素时有更多控制的数据结构。有两种数据结构类似于数组,但在添加和删除元素时更为可控。它们就是栈和队列。​ 要开始学......
  • redis 数据结构
    数据结构预算法最难啃,并且redis底层是c,需要熟悉c才好根据源码分析。先占坑吧SDSredis的String的数据结构,全称为简单动态字符串,simpledynamicstring,redis是c编写的,为什么不用c语言的字符串类型呢,肯定是为了优化性能而自定义的一种数据类型举个简单的例子:c获取字符串......
  • 数据结构专题 1
    图论狗都不写。宁可写数据结构也不想写图论了。写吐了。牛子老师说这套题的后半全是正经数据结构,而且无Ynoi。所以啥时候开多项式。由于写题解主要是合集,因此打算分拆一下水点社贡。目前停留在打算阶段。日,为什么明天考试。CF1039DYouAreGivenaTree很久以前看到过。这......
  • 数据结构期末复习——图的遍历
    图的遍历:1.定义:从某个结点出发访问遍图中结点,且使每个结点仅被访问一次图的遍历具有复杂性,主要体现在以下几点1.遍历没有规定从哪个结点开始访问,因此从任意结点开始访问均可2.图的一个结点可以连接多个结点,因此无法确定访问此结点之后应该访问哪一个结点3.如果一个图中存在回......
  • 5_24_打卡_数据结构之循环队列
    //循环队列可存储数据数量是maxsize-1//队列长度为(front-rear+maxsize)%maxsize//队列为空时front==rear//队列满时(front+1)%maxsize==rear;#defineMAXSIZE5#include<iostream>usingnamespacestd;typedefstructqueue{ intfront; intrear; intdata[MAXSIZE];}......
  • 关于redis的描述、数据结构、持久化学习笔记
    前言本文围绕面试问题、redis学习记录。本文是个人的笔记,会有遗漏或含糊的地方。描述下redisredis是一款非关系型数据库,它是以key-value的形式存在数据,因为它的数据在内存中所以它的读写速度极高。当然它支持持久化,将数据以二进制形式或者以命令的形式持久化到磁盘。然后......