堆栈

✍ dations ◷ 2025-06-09 02:01:06 #数据结构,抽象数据类型

堆栈(英语:stack)又称为栈或堆叠,是计算机科学中的一种抽象数据类型,只允许在有序的线性数据集合的一端(称为堆栈顶端,英语:top)进行加入数据(英语:push)和移除数据(英语:pop)的运算。因而按照后进先出(LIFO, Last In First Out)的原理运作。

常与另一种有序的线性数据集合队列相提并论。

堆栈常用一维数组或链表来实现。

堆栈使用两种基本操作:推入(压栈,push)和弹出(弹栈,pop):

堆栈的基本特点:

以下是堆栈的VDM():

函数签名:

  init: -> Stack  push: N x Stack -> Stack  top: Stack -> (N U ERROR)  pop: Stack -> Stack  isempty: Stack -> Boolean

此处的N代表某个元素(如自然数),而U表示集合求交。

语义:

  top(init()) = ERROR  top(push(i,s)) = i  pop(init()) = init()  pop(push(i, s)) = s  isempty(init()) = true  isempty(push(i, s)) = false

软件堆栈

堆栈可以用数组和链表两种方式实现,一般为一个堆栈预先分配一个大小固定且较合适的空间并非难事,所以较流行的做法是Stack结构下含一个数组。如果空间实在紧张,也可用链表实现,且去掉表头。

这里的例程是以C语言实现的。

 1 #include <stdio.h> 2 #include <stdlib.h> 3 #include <stdbool.h> 4 #define stack struct Stack 5 #define STACK_POP_ERR 42 6  7 // 堆疊資料結構 堆栈数据结构 8 struct Stack { 9     int val; // 陣列空間10     int top;     // 堆疊頂端指標(栈顶)11 };12 // 檢查堆疊是否為空13 bool empty(stack *stk) {14     return stk->top == 0;15 }16 // 推入資料17 void push(stack *stk, int x) {18     stk->top = stk->top + 1;19     stk->val = x;20 }21 // 彈出并返回資料22 int pop(stack *stk) {23     if (empty(stk))24         return STACK_POP_ERR; // 不能彈出25     else {26         stk->top = stk->top - 1;27         return stk->val;28     }29 }30 int main() {31     // 宣告并初始化資料結構空間32     stack stk;33     stk.top = 0;34     // 推入四个35     push(&stk, 3);36     push(&stk, 4);37     push(&stk, 1);38     push(&stk, 9);39     // 弹出三个40     printf("%d ", pop(&stk));41     printf("%d ", pop(&stk));42     printf("%d ", pop(&stk));43     return 0;44 }

串列堆栈

  1 #include <stdio.h>  2 #include <conio.h>  3 #include <stdlib.h>  4   5 #define elemType int							/* 链栈元素数据类型,以整型为例 */  6 #define SNODE_SIZE sizeof (struct sNode)		/* 链栈结点空间大小 */  7   8 #define status int	/* 状态型变量 */  9 #define OVERFLOW -1	/* 内存溢出状态码 */ 10 #define ERROR 0		/* 错误状态码 */ 11 #define OK 1		/* 正确状态码 */ 12  13 /* 链栈结点存储结构 */ 14 typedef struct sNode { 15 	elemType data; 16 	struct sNode *next; 17 } sNode, *sNodePtr; 18  19 /* 链栈存储结构 */ 20 typedef struct linkStack { 21 	sNodePtr top; /* 栈顶指针 */ 22 } linkStack; 23  24 /* 初始化 */ 25 /* 操作结果:构造一个带头结点的空链栈S */ 26 void initStack (linkStack *S) { 27 	S->top = (sNodePtr) malloc (SNODE_SIZE); /* 产生头结点,栈顶指针指向此头结点 */ 28 	if (!S->top) /* 内存分配失败 */ 29 		exit (OVERFLOW); 30 	S->top->next = NULL; 31 } 32  33 /* 销毁 */ 34 /* 初始条件:链栈S已存在。操作结果:销毁链栈S */ 35 void destroyStack (linkStack *S) { 36 	sNodePtr p, q; 37 	 38 	p = S->top; /* p指向S的头结点 */ 39 	while (p) { 40 		q = p->next; /* q指向p的下一个结点 */ 41 		free (p); /* 回收p指向的结点 */ 42 		p = q; /* p移动到下一个结点 */ 43 	} /* 直到没有下一个结点 */ 44 } 45  46 /* 清空 */ 47 /* 初始条件:链栈S已存在。操作结果:将S重置为空栈 */ 48 void clearStack (linkStack *S) { 49 	sNodePtr p, q; 50 	 51 	p = S->top->next; /* p指向栈的第一个结点 */ 52 	while (p) { 53 		q = p->next; /* q指向p的下一个结点 */ 54 		free (p); /* 回收p指向的结点 */ 55 		p = q; /* p移动到下一个结点 */ 56 	}  /* 直到没有下一个结点 */ 57 	 58 	S->top->next = NULL; 59 } 60  61 /* 判断链栈是否为空 */ 62 /* 初始条件:链栈S已存在。操作结果:若S为空链栈,则返回TRUE,否则返回FALSE */ 63 status stackIsEmpty (linkStack *S) { 64 	return S->top->next == NULL; 65 } 66  67 /* 求链栈长度 */ 68 /* 初始条件:链栈S已存在。操作结果:返回S中数据元素个数 */ 69 int stackLength (linkStack *S) { 70     int i = 0; 71     sNodePtr p; 72 	 73 	p = S->top->next; /* p指向栈的第一个结点 */ 74 	while (p) { /* 未到栈尾 */ 75 		i++; 76 		p = p->next; 77     } 78      79     return i; 80 } 81  82 /* 获取栈顶元素 */ 83 /* 初始条件:链栈S已存在。操作结果:当栈不为空时,将栈顶元素其值赋给e并返回OK,否则返回ERROR */ 84 status getTopElem (linkStack *S, elemType *e) { 85 	sNodePtr p; 86 	 87 	if (stackIsEmpty (S)) 88 		return ERROR; 89 	 90 	p = S->top->next; /* p指向栈的第一个结点 */ 91 	*e = p->data; 92 	 93 	return OK; 94 } 95  96 /* 入栈 */ 97 /* 操作结果:在S的栈顶插入新的元素e */ 98 status push (linkStack *S, elemType e) { 99 	sNodePtr p;100 	101 	p = (sNodePtr) malloc (SNODE_SIZE); /* 产生新结点 */102 	if (!p) /* 内存分配失败 */103 		exit (OVERFLOW);104 	105 	p->data = e;106 	p->next = S->top->next; /* 将新结点链接到原栈顶 */107 	S->top->next = p; /* 栈顶指向新结点 */108 }109 110 /* 出栈 */111 /* 操作结果:删除S的栈顶元素,并由e返回其值 */112 status pop (linkStack *S, elemType *e) {113 	sNodePtr p;114 	115 	if (stackIsEmpty (S))116 		return ERROR;117 	118 	p = S->top->next; /* p指向链栈的第一个结点 */119 	*e = p->data; /* 取出数据 */120 	S->top->next = p->next;121 	free (p); /* 删除该结点 */122 	123     if (S->top == p) /* 栈为空 */124     	S->top->next = NULL;125     126     return OK;127 }128 129 /* 打印栈内容 */130 /* 初始条件:链栈S已存在。操作结果:当栈不为空时,打印栈内容并返回OK,否则返回ERROR */131 status printStack (linkStack *S) {132 	sNodePtr p;133 	134 	if (stackIsEmpty (S))135 		return ERROR;136 	137 	p = S->top->next;138 	while (p) {139 		printf ("%d\t", p->data);140 		p = p->next;141 	}142 	putchar ('\n');143 	144 	return OK;145 }

堆栈有时候也常用来指代堆栈段。

架构层次上的堆栈通常被用以申请和访问内存。

大多数CPU都有用作堆栈指针的寄存器。

相关

  • 专利蟑螂专利流氓又称专利蟑螂(英语:Patent Troll),用于形容一些积极发动专利侵权诉讼以获取赔偿,却从没生产其专利产品的个人或公司。至2008年为止美国是全世界专利纠纷最多的国家之一,其
  • 顶体 (精子)顶体(英语:acrosome,来自希腊语:acron (外面)和soma (体))是动物精子核前部由高尔基体所衍生来的顶状膜包结构。在真兽下纲的哺乳动物中,顶体含有多种酶(包括玻璃酸酶(英语:Hyaluroni
  • 托多尔·日夫科夫托多尔·日夫科夫(保加利亚语:Toдор Xpиcтoв Живков,IPA:;1911年9月7日-1998年8月5日),保加利亚政治人物。1954年-1989年,担任保加利亚共产党中央委员会第一书记,也担任保
  • 佛经翻译佛经翻译指的是将佛经从梵语或其它西域语言翻译成汉语的程序。亦中国翻译史上的一件大事,所翻译出的佛经对中华文明产生巨大影响及贡献。佛经翻译事业开始于东汉,至前秦苻坚开
  • 浊硬颚内爆音浊硬颚内爆音(英语:)是一个用于部分语言的辅音,见于南亚及西非。用来表示这个音的IPA符号是:⟨ ʄ ⟩,即一个无点j加上删节线(即浊硬颚塞音的符号)后再在上面加用来表示内爆音的右
  • 中都镇 (上杭县)中都镇是中国福建省龙岩市上杭县下辖的一个镇。宋朝时,中都为上杭七里之一的来苏里。明末清初,上杭胜运里的绅士在兴市(今中都镇墟场)南面修建集市“瓜墩墟”。清乾隆、嘉庆年间
  • 马克斯·胡贝尔马克斯·胡贝尔(德语:Max Huber;1874年12月28日-1960年1月1日)是一位瑞士律师、外交家,代表瑞士参加一系列国际会议与机构。胡贝尔生于苏黎世,曾在洛桑大学、苏黎世大学和柏林大学
  • 十二金刚《十二金刚》(英语:)是一套1967年首映的美国战争片,剧情改编自E·M·奈森桑(E.M. Nathanson)的同名小说。此片由罗伯特·阿德烈治执导,李·马文、喧尼斯·鲍宁、查尔斯·布朗森、尊
  • KompoZerKompoZer是一个开放源代码的所见即所得HTML编辑器,它基于已经停止开发的Nvu编辑器。 KompoZer项目现在托管在SourceForge,由社区负责维护。KompoZer的主要功能是所见即所得地
  • 反例在逻辑学中,反例是相对于某个全称命题的概念。反例在数学、哲学和自然科学中都有重要的应用。举例来说,对一个命题:所有的天鹅都是白色的。这是一个全称命题,声明对于某类事物全