手撕数据结构—栈
创始人
2025-05-31 02:53:51
0

Tips

  1. 不得不再次提一下这个语法问题,当数组创建的时候,进行初始化的时候,分为全部初始化或者说部分初始化,对于不完全初始化而言,剩下的部分就全部默认为零。现在比如说你想对整型数组的1万个元素把它全部变成-1,不能够仅仅在一个花括号里面写个-1,只是第一个元素变成-1,然后其他的都变成0了。之后你只能用memset

栈以及先进后出原则

  1. 栈和队列其实也是一个线性表。线性表也就是说你这个数据至少在逻辑上都是依次线性存储,一个一个挨着挨着这样存储这么一个概念。

  1. 栈作为一种特殊的线性表,它只允许在固定的一端进行数据的插入或删除元素操作。进行数据插入和删除操作的那一端就被称为栈顶。因此很容易理解栈中的数据元素遵守后进先出原则。

压栈与出栈

  1. 栈的插入操作叫做进栈/压栈/入栈,这是在栈顶完成的。

  1. 栈的删除操作叫做出栈,这也是在栈顶完成的。所以说它是在同一端进行操作。

  1. 在这边值得一提的是,比如说现在有一堆元素,对于同一进栈的顺序,但是出栈序列可以多种多样,因为并没有规定什么时候可以出栈,你可以使所有元素放进栈里面之后再依次出栈;当然你也可以是在边进栈的过程中可以出栈。我随便举个例子好了:比如说进栈序列为1234,那么出栈序列可以比如为1432,2341,3421,但是断然断然不可能是3124。


栈的实现

  1. 栈的话可以用数组去实现,也可以用链表去实现。肯定是数组,用数组来实现栈的话嘎嘎香啊,比如说你就可以把数组的右端当成一个栈的栈顶。如果要说真有一个弊端的话,那就是说用数组来模拟栈的话需要扩容。

  1. 那如果非要用链表去实现,也是完全可以的:能用单链表就用单链表,你用双向链表的话,还多一个指针呢,能省一点就是一点。

  1. 但是用单链表的话由于尾删啊尾插啊这些操作都需要去从phead开始去往后去遍历去找尾(注意链表不支持下标访问操作),这会相当的麻烦,因此就想了一个办法。把整个链表的左端当成栈顶,那么这样子的话,我的入栈与出栈相当于单链表的头插头删,效率非常之高。

  1. 但如果说非要选一个的话,用数组和链表来模拟的话都非常可以,因为都是O(1)的插入删除(数组的话可以支持下标访问,把数组的右端当成栈顶;链表的话把他的左端当成栈顶,头插头删也是O(1))。可能你还是会选择链表,但是别忘了数组的缓存命中率与利用率比链表要高。


栈的创建(创建结构体)

  1. 凡是有多个数据,都放到一个结构体里面。

  1. 对于这个结构体有三个要素,一个是容量,一个是栈顶top,还有一个我是等会儿从堆区开辟内存空间之后返回来的地址,需要用一个指针接收一下,标记一下地址。

typedef int STDataType;
typedef struct Stack
{STDataType* p;int top;int capacity;
}ST;

栈的初始化

  1. 在初始化的时候有一个比较容易出错的地方,就是必须得先搞清楚这个top到底是什么东西,我就假定这个top指向此时此刻的栈顶元素。那么这时候由于要初始化,此时栈顶也压根儿没有任何元素,因此top就指向-1/那如果我说这个top是栈顶元素的下一个位置,那此时此刻初始化的时候,这个top应该指向0。

  1. 我们为了跟之前的顺序表保保持一致,初始化的时候,这个top就给他弄成0。此时此刻,你只需要记住top的一个含义:他现在就表示栈中的元素个数

#define INIT_CAPACITY 5
void STInit(ST* ps)
{assert(ps);ps->p = (STDataType*)malloc(sizeof(STDataType) * INIT_CAPACITY);if (ps->p == NULL){perror("STInit::Malloc");return;}ps->top = 0;ps->capacity = INIT_CAPACITY;
}

栈的销毁

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

入栈

void STPush(ST* ps, STDataType x)
{assert(ps);if (ps->top == ps->capacity){STDataType* pp = (STDataType*)realloc(ps->p, sizeof(STDataType) * (ps->capacity) * 2);if (pp == NULL){perror("STPush::Realloc");return;}ps->p = pp;ps->capacity *= 2;}ps->p[ps->top] = x;ps->top++;
}

栈的判断是否为空

bool STEmpty(ST* ps)
{assert(ps);return ps->top==0;
}

栈的求元素个数

int STSize(ST* ps)
{assert(ps);return ps->top;
}

出栈

void STPop(ST* ps)
{assert(ps);assert(!STEmpty(ps));ps->top--;
}

栈的求栈顶元素

int STTop(ST* ps)
{assert(ps);assert(!STEmpty(ps));return ps->p[ps->top - 1];
}

注:

  1. 虽然从代码上看起来与顺序表非常非常的相像。但是栈的话一定要记住他的特性,那就是后进先出。比如说:23458,我如果要访问5,那么8就必须先出去,如果说我要访问3,那么458就必须先出去。正是因为这种后进先出的特性,这也导致了我们没有写打印栈这种函数,因为栈这种玩意儿,他是不支持去遍历的,这是规定死的。

  1. 这些都是由栈的性质决定的,否则他就不叫做栈了。对于先进栈的数据,想要对他进行任何的操作,包括访问与打印,都必须把它之前栈顶的元素全部弹出去才可以,不然永远只能对栈顶的那个元素动手。

相关内容

热门资讯

最新或2023(历届)小学国庆...   最新或2023(历届)小学国庆节活动方案策划书一:  一、活动目的:  为隆重、热烈、活泼的庆祝...
超市国庆如何做促销,超市国庆节...   第一部分:各店促销活动内容:  一、百货区买200送200  超市区好礼大赠送感恩2+1卡免费办...
最新或2023(历届)国庆节超...   现场散发本超市十一活动的清单,印制彩页,彩页上附抵用券,超市知识有奖问答 。  (五)慢骑比赛 ...
小学最新或2023(历届)国庆...   小学最新或2023(历届)国庆庆祝活动方案  活动目的:通过本次庆祝活动,培养中兴小学少先队员民...
商场国庆节活动方案,资料大全 ...   XX年9月23日——XX年10月8日7号就结束了国庆长假,而中秋送礼,吃团圆饭更是在6号之前的事...
怎样进行Redis客户端的设置... 安装完成Redis,我们就可以操作Redis,实现数据的CRUD了。这需...
AIOT处理平台RK3568开... Rockchip RK3568 处理器是一款高性能、低功耗的四核应用处理器芯片,专为个...
商场国庆活动策划方案,资料大全...   随着中秋节和国庆节的临近,各大商场都是攒足了劲儿,拼了命的要在这个黄金周中独领风骚,为此,龙霞唯...
如何通过企业小程序开发提升企业... 随着移动互联网的普及,企业小程序已成为企业进行线上营销和服务的重要工具之一。如何通过企...
小学生迎国庆活动方案及主持人串...   刘:从金黄的季节走来,国庆节是一首优美的诗歌;卢:从历史的蜿蜒走来,国庆节是一抹光辉的记忆;刘:...
最近小学国庆节活动方案,资料大...   小学国庆节活动方案:国庆活动方案  一、 活动背景  少先队是我们少年儿童的先锋队,其“文化育人...
如何使用租用的云服务器实现神经... B站教学视频:https://www.bilibili.com/video/BV1jA...
各个商场促销活动总汇,资料大全...   国庆黄金周快到了,昨日,各大百货商场纷纷抛出了节日促销信息。记者挨着逛了一圈,替市民“绘制”了一...
商城国庆节促销活动方案,“新”...   活动时间:xx月xx日(周五)——xx月xx日(周日)  活动范围:某商场商城及八一店、某商场购...
最新或2023(历届)商场国庆...   一、活动目的:中秋节的活动已经结束,由于人们在节日期间走亲串友,家中的礼品类商品比较充足,因此在...
小红书图文笔记怎么发?小红书笔...   说到图文笔记,这可是小红书的一大利器。从大方向来说,这是小红书成功从...
数字中国看“浙”里丨太平鸟、实... 当前,数字经济已成为重组全国要素资源、变革经济格局的关键力量。中共中央、国务院印发的《...
Mysql的主从复制原理 MySQL 的主从复制依赖于 binlog ,也就是记录 MySQL 上的所有变化并以...
小学国庆节活动方案策划,庆祝国...   活动目的:庆祝国庆节  通过本次庆祝活动,培养队员热爱祖国,了解祖国的悠久历史,培养民族自豪感。...
最新或2023(历届)国庆节活...   活动目的:通过本次庆祝活动,培养中兴小学少先队员民族自豪感,调动每个学生积极参与的热情,真切感受...