数据结构(三)--栈
数据结构(三)–栈
通常程序开发中内存管理是非常重要的,而内存主要分为占内存和堆内存。那么栈和堆内存有什么区别呢?希望在这篇文章里能带你找到答案!
1. 栈和堆的引入
在一个简单的程序中我们定义和声明几个基本类型的变量、结构体和数组,先来直观看一下栈和堆的不同:
- 静态变量 和 局部变量是以压栈出栈的方式分配内存的,系统会在一个代码段中分配和回收局部变量,实际上每个代码段、函数都是一个或多个嵌套的栈,我们不需要手动管理栈区内存。
- 动态内存是一种堆排序的方式分配内存的,内存分配好后便不会自动回收,需要程序员手动回收。否则就会造成内存泄漏,内存越用越少。
简单了解了一下程序中内存栈与堆的区别,下面就正式开始讲数据结构中的栈。
(注意:数据结构栈、内存栈、函数调用栈三者在含义上略有不同,但是其核心思想和理念是相同的)
2. 栈的定义
栈是一种“先进后出”的一种数据结构,有压栈出栈两种操作方式。如下图:
3. 栈的分类
栈主要分为两类:
- 静态栈
- 动态栈
【静态栈】
静态栈的核心是数组,类似于一个连续内存的数组,我们只能操作其栈顶元素。
【动态栈】
静态栈的核心是数组,类似于一个连续内存的数组,我们只能操作其栈顶节点。
4. 栈的算法
栈的算法主要是压栈和出栈两种操作的算法,下面我就用代码来实现一个简单的栈。
首先要明白以下思路:
- 栈操作的是一个一个节点
- 栈本身也是一种存储的数据结构
- 栈有
初始化
、压栈
、出栈
、判空
、遍历
、清空
等主要方法
4.1 栈的头文件定义
头文件定义如下:
typedef struct Node{ // 节点
int data;
struct Node *pNext;
}*PNODE,NODE;
typedef struct Stack{ // 栈
PNODE pTop;
PNODE pBottom;
}STACK,*PSTACK;
/**栈的初始化*/
void init(PSTACK);
/**压栈*/
void push(PSTACK,int);
/**出栈*/
int pop(PSTACK , int *);
/**遍历打印栈*/
void traverse(PSTACK);
/**是否为空栈*/
int isEmpty(PSTACK);
/**清空栈*/
void clearStack(PSTACK);
有了头文件定义,基本就确定了栈的使用结构和使用方式。下面是在主函数中对这个栈的创建和使用。
int main(void){
STACK stack; // 声明一个栈
init(&stack); // 初始化
// 压栈
push(&stack, 10);
push(&stack, 20);
push(&stack, 30);
push(&stack, 40);
push(&stack, 50);
traverse(&stack); // 遍历打印栈
int val;
int isPopSuccess = pop(&stack,&val);
if (isPopSuccess) {
printf("pop 的值为 %d\n",val);
}
traverse(&stack);
clearStack(&stack); // 清空栈
traverse(&stack);
return 0;
}
4.2 栈的初始化
思路:
拿到栈声明的指针,开辟一块内存空间给栈顶栈底,此时是一个空栈,栈顶栈底指向同一块内存,且栈底栈顶以外不再指向其他节点。
/**栈的初始化*/
void init(PSTACK pS){
pS->pTop = (PNODE)malloc(sizeof(NODE));
if (pS->pTop == NULL) {
printf("内存分配失败退出");
return;
}else
{
pS->pBottom = pS->pTop;
pS->pTop->pNext = NULL;
}
}
4.3 压栈 和 出栈
思路:
压栈是把新的节点放入栈顶,且每次压栈操作只能将新的节点放到栈的顶部。
出栈需判断是否原本为空栈,存在出栈失败的情况,把栈顶指向栈顶元素的下一个元素,并释放原来栈顶元素空间。
/**
压栈
@param pS 执行压栈的栈指针
@param val 被压栈的值
*/
void push(PSTACK pS,int val){
// 创建新节点,放到栈顶
PNODE pNew = (PNODE)malloc(sizeof(NODE));
pNew->data = val;
pNew->pNext = pS->pTop;
pS->pTop = pNew; // 栈顶指针指向新元素
}
/**
出栈
@param pS 执行出栈的栈地址
@param val 出栈值的地址
@return 是否出栈成功
*/
int pop(PSTACK pS , int *val){
if (isEmpty(pS)) {
printf(" 空栈 ,出栈失败");
return 0;
}else
{
PNODE p = pS->pTop;
pS->pTop = p->pNext;
if (val != NULL) {
*val = p->data;
}
free(p); // 释放原来top内存
p = NULL;
return 1;
}
}
/**是否为空栈*/
int isEmpty(PSTACK pS)
{
if (pS->pTop == pS->pBottom) {
return 1;
}else
{
return 0;
}
}
4.4 栈的清空 和 遍历
当一个代码段执行完成之后,实际上就是这个栈所有分配的空间都被回收,栈随之被清空!
思路:
栈清空,实际就是需要循环执行出栈操作。
栈遍历,实际就是栈元素从栈顶一个个遍历到栈底,可以打印栈中元素的值
/**清空栈*/
void clearStack(PSTACK pS){
if (isEmpty(pS)) {
return;
}else{
PNODE p = pS->pTop;
PNODE q = NULL;
while (p!=pS->pBottom) {
q = p->pNext;
free(p); // 释放原栈顶元素空间
p = q;
}
pS->pTop = pS->pBottom;
}
// 偷懒的做法
// while (!isEmpty(pS)) {
// pop(pS, NULL);
// }
}
/**遍历打印栈*/
void traverse(PSTACK pS){
// 只要不是空栈,就一直输出
PNODE p = pS->pTop;
while (p != pS->pBottom) {
printf("%d ",p->data);
p = p->pNext; // 把top的下一个节点付给top,继续遍历
}
printf("\n");
}
5. 栈的应用
栈结构固有的先进后出的特性,使它成为在程序设计中非常有用的工具,这里列举几个典型的例子。
5.1 数制转换
十进制数 N 和其他 d 进制数的转换是计算机实现计算的基本问题,其解决方法有很多种,其中一个简单的方法基于如下原理:N = (N div d) * d + N mod d
(其中div是整除运算,mod 为求余运算)
例如:1348(10进制) == 2504(8进制)运算过程如下:
N | N div 8 | N mod 8 |
---|---|---|
1348 | 168 | 4 |
168 | 21 | 0 |
21 | 2 | 5 |
2 | 0 | 2 |
需求:输入一个任意非负十进制整数,打印输出其对应的八进制整数
思路:由于上述计算过程是从低到高位顺序产生八进制数的各个数位,而打印输出,一般来说应从高位到低位进行,恰好和计算过程相反。因此可利用栈先进后出特性,将计算过程中得到的八进制数各位顺序进栈,再按出栈序列打印输出既为与输入对应的八进制数。
void conversion(void){
// 创建栈
STACK S;
init(&S);
// 用户输入十进制数
scanf("%d",&N);
// 放入栈中
while (N) {
push(&S, N % 8);
N = N / 8;
}
// 打印出来
printf("对应八进制数字为:");
int a;
while (!isEmpty(&S)) {
pop(&S, &a);
printf("%d",a);
}
printf("\n");
}
思考 用数组实现貌似更简单,为什么不用数组?
从算法上分析不难看出,栈的引入简化了程序设计的问题,划分了不同的关注层次,使思考范围缩小了。而使用数组不仅掩盖了问题的本质,还要分散精力去思路数组下标增减等细节问题。
这也是早期面向对象编程的一种思想,要把对应的功能划分关注层次,在逻辑的实现上面更加专注问题的本质。
5.2 括号匹配的检验
编程语言中基本都允许使用 (),[],{}
这几种括号,假设现在让使用两种,一段完整代码中其须成对匹配,检验括号是否匹配的方法可用”期待的紧迫程度”这个概念来描述。
当计算机接受了第一个括号后,它期待着与其匹配的第八个括号出现,然而等来的确实第二个括号,此时第一个括号[
只能暂时靠边,而迫切等待与第二个括号匹配的第七个括号)
出现,类似地,等来的是第三个括号[
,其期待的匹配程度比第二个更加急迫,则第二个括号也只能靠边,让位于第三个括号,显然第二个括号的期待急迫性高于第一个括号,在接受了第四个括号之后,第三个括号的期待得到满足,消解之后,第二个括号的期待匹配变成最紧迫的任务了·····,以此类推。
可见此处理过程与栈的特点相吻合,由此,在算法中设置一个栈,每读入一个括号,若是右括号则使至于栈顶的最紧迫的期待得以消解,若是不合法的情况(左括号),则作为一个新的更紧迫的期待压入栈中,自然使原来所有未消解的期待的紧迫性都降了一级。另外在算法开始和结束的时候,栈都应该是空的。
算法实现:
/**
检测括号(本实例用数字代替括号)
[ ] --> 1 , 2
( ) --> 3 , 4
*/
void checkBracelet(void)
{
// 创建栈
STACK S;
init(&S);
// 用户输入括号
int N;
printf("请输入对应的括号(end结束):\n");
scanf("%d",&N);
if (isEmpty(&S)) {
push(&S, N);
printf("第一个括号输入\n");
traverse(&S); // 打印此时栈内容
}
while (!isEmpty(&S)) {
// 用户输入括号
int N;
printf("请输入对应的括号(0结束):\n");
scanf("%d",&N);
if (N == 0) {
break; // 用户输入0直接退出
}
// 判断当前栈顶是否符合标准,
if (S.pTop->data == N) {
printf("消除一对\n");
pop(&S, NULL);
traverse(&S); // 打印此时栈内容
}else
{
printf("未消除\n");
push(&S, N);
traverse(&S); // 打印此时栈内容
}
}
}
这里的实例我列举了两个,实际上还有很多。比如 行编辑程序、迷宫求解、表达式求值等。这里我就先不做列举了。
6. 小结
通过这里复习数据结构中栈的内容,感觉重新理解了很多计算机实现的底层知识,虽然不知道的更多,但是面对计算机心中又多了一层认知!
html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,big,cite,code,del,dfn,em,img,ins,kbd,q,s,samp,small,strike,strong,sub,sup,tt,var,b,u,i,center,dl,dt,dd,ol,ul,li,fieldset,form,label,legend,table,caption,tbody,tfoot,thead,tr,th,td,article,aside,canvas,details,embed,figure,figcaption,footer,header,hgroup,menu,nav,output,ruby,section,summary,time,mark,audio,video { margin: 0; padding: 0; border: 0; font: inherit; font-size: 100%; vertical-align: baseline }
html { line-height: 1 }
ol,ul { list-style: none }
table { border-collapse: collapse; border-spacing: 0; margin-top: 0; margin-bottom: 0.8em }
caption,th,td { text-align: left; font-weight: normal; vertical-align: middle }
q,blockquote { quotes: none }
q::before,q::after,blockquote::before,blockquote::after { content: none }
a img { border: none }
article,aside,details,figcaption,figure,footer,header,hgroup,menu,nav,section,summary { display: block }
a { color: #1863a1 }
a:visited { color: #751590 }
a:focus { color: #0181eb }
a:hover { color: #0181eb }
a:active { color: #01579f }
aside.sidebar a { color: #222 }
aside.sidebar a:focus { color: #0181eb }
aside.sidebar a:hover { color: #0181eb }
aside.sidebar a:active { color: #01579f }
a { }
body,h1,h2,h3,h4,h5,h6,footer { font-family: “PT Sans”, “Helvetica Neue”, “Optima”, “Hiragino Sans GB”, sans-serif }
body { line-height: 1.5em; color: #222 -webkit-text-size-adjust:none; min-width: 200px; max-width: 760px; margin: 0 auto; padding: 1em }
pre,code,tt,p code,li code { font-family: Menlo, Monaco, “Andale Mono”, “lucida console”, “Courier New”, monospace }
h1 { font-size: 2.2em; line-height: 1.2em }
h1,h2,h3,h4,h5,h6 { margin-bottom: 1em; font-weight: bold }
h2,section h1 { font-size: 1.5em }
h3,section h2,section section h1 { font-size: 1.3em }
h4,section h3,section section h2,section section section h1 { font-size: 1em }
h5,section h4,section section h3 { font-size: .9em }
h6,section h5,section section h4,section section section h3 { font-size: .8em }
.markdown-body { padding: 0px 4px }
.markdown-body h1 { position: relative; padding-top: 1em; padding-bottom: 0.2em; margin-bottom: 1em; background: url(“”) bottom left repeat-x }
.markdown-body h1 a { text-decoration: none }
.markdown-body h1 a:hover { text-decoration: underline }
.markdown-body h2 { padding-top: 0.8em; padding-bottom: 0.2em; background: url(“”) bottom left repeat-x }
.markdown-body h2:first-child,.markdown-body header+h2 { padding-top: 4px }
.markdown-body h2:first-child,.markdown-body header+h2 { background: none }
p,.markdown-body blockquote,ul,ol { margin-bottom: 0.8em; margin-top: 0.8em }
ul { list-style-type: disc }
ul ul { list-style-type: circle; margin-bottom: 0px }
ul ul ul { list-style-type: square; margin-bottom: 0px }
ol { list-style-type: decimal }
ol ol { list-style-type: lower-alpha; margin-bottom: 0px }
ol ol ol { list-style-type: lower-roman; margin-bottom: 0px }
ul,ul ul,ul ol,ol,ol ul,ol ol { margin-left: 1.3em }
ul ul,ul ol,ol ul,ol ol { margin-bottom: 0em }
strong { font-weight: bold }
em { font-style: italic }
sup,sub { font-size: 0.75em; position: relative; display: inline-block; padding: 0 .2em; line-height: .8em }
sup { top: -.5em }
sub { bottom: -.5em }
a[rev=”footnote”] { font-size: .75em; padding: 0 .3em; line-height: 1 }
q { font-style: italic }
q::before { content: ““” }
q::after { content: “”” }
em,dfn { font-style: italic }
strong,dfn { font-weight: bold }
del,s { text-decoration: line-through }
abbr,acronym { border-bottom: 1px dotted; cursor: help }
small { font-size: .8em }
big { font-size: 1.2em }
.markdown-body hr { height: 0; margin: 15px 0; overflow: hidden; background: transparent; border: 0; border-bottom: 1px solid #ddd }
.markdown-body hr::before { display: table; content: “” }
.markdown-body hr::after { display: table; clear: both; content: “” }
.markdown-body table { display: block; width: 100%; overflow: auto }
.markdown-body table th { font-weight: bold }
.markdown-body table th,.markdown-body table td { padding: 6px 13px; border: 1px solid #ddd }
.markdown-body table tr { background-color: #fff; border-top: 1px solid #ccc }
.markdown-body table tr:nth-child(2n) { background-color: #f8f8f8 }
.markdown-body blockquote { font-style: italic; position: relative; font-size: 1.2em; line-height: 1.5em; padding-left: 1em; border-left: 4px solid rgba(170,170,170,0.5) }
.markdown-body blockquote cite { font-style: italic }
.markdown-body blockquote cite a { color: #aaa !important }
.markdown-body blockquote cite::before { content: “—”; padding-right: .3em; padding-left: .3em; color: #aaa }
.markdown-body a { white-space: pre-wrap }
body>header { font-size: 1em; padding-top: 1.5em; padding-bottom: 1.5em }
.markdown-body { overflow: hidden }
.markdown-body>div,.markdown-body>article { width: 100% }
aside.sidebar { float: none; padding: 0 18px 1px; background-color: #f7f7f7; border-top: 1px solid #e0e0e0 }
.flex-content,article img,article video,article .flash-video,aside.sidebar img { max-width: 100%; height: auto }
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left { float: left; margin-right: 1.5em }
.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right { float: right; margin-left: 1.5em }
.basic-alignment.center,article img.center,article video.center,article .center.flash-video,aside.sidebar img.center { display: block; margin: 0 auto 1.5em }
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left,.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right { margin-bottom: .8em }
.toggle-sidebar,.no-sidebar .toggle-sidebar { display: none }
.markdown-body img,.markdown-body video,.markdown-body .flash-video { border: #fff 0.5em solid }
.markdown-body img,.markdown-body video { max-width: 100% }
.markdown-body video,.markdown-body .flash-video { margin: 0 auto 1.5em }
.markdown-body video { display: block; width: 100% }
.markdown-body .flash-video>div { position: relative; display: block; padding-bottom: 56.25%; padding-top: 1px; height: 0; overflow: hidden }
.markdown-body .flash-video>div iframe,.markdown-body .flash-video>div object,.markdown-body .flash-video>div embed { position: absolute; top: 0; left: 0; width: 100%; height: 100% }
.markdown-body>footer { padding-bottom: 2.5em; margin-top: 2em }
.markdown-body>footer p.meta { margin-bottom: .8em; font-size: .85em; clear: both; overflow: hidden }
body,pre { background: #fdf6e3 url(“”) top left }
body { background-color: #f8f8f8 }
pre { border: 1px solid #e7dec3; line-height: 1.45em; font-size: 13px; margin-bottom: 2.1em; padding: .8em 1em; color: #586e75; overflow: auto }
.markdown-body code { background: none }
h3.filename+pre { }
p code,li code { display: inline-block; white-space: no-wrap; background: #fff; font-size: .8em; line-height: 1.5em; color: #555; border: 1px solid #ddd; padding: 0 .3em; margin: -1px 0 }
p pre code,li pre code { font-size: 1em !important; background: none; border: none }
.hljs { display: block; padding: 0.5em; background: #fdf6e3; color: #657b83 }
.hljs-comment,.diff .hljs-header,.hljs-doctype,.hljs-pi,.lisp .hljs-string { color: #93a1a1 }
.hljs-keyword,.hljs-winutils,.method,.hljs-addition,.css .hljs-tag,.hljs-request,.hljs-status,.nginx .hljs-title { color: #859900 }
.hljs-number,.hljs-command,.hljs-string,.hljs-tag .hljs-value,.hljs-rule .hljs-value,.hljs-doctag,.tex .hljs-formula,.hljs-regexp,.hljs-hexcolor,.hljs-link_url { color: #2aa198 }
.hljs-title,.hljs-localvars,.hljs-chunk,.hljs-decorator,.hljs-built_in,.hljs-identifier,.vhdl .hljs-literal,.hljs-id,.css .hljs-function,.hljs-name { color: #268bd2 }
.hljs-attribute,.hljs-variable,.lisp .hljs-body,.smalltalk .hljs-number,.hljs-constant,.hljs-class .hljs-title,.hljs-parent,.hljs-type,.hljs-link_reference { color: #b58900 }
.hljs-preprocessor,.hljs-preprocessor .hljs-keyword,.hljs-pragma,.hljs-shebang,.hljs-symbol,.hljs-symbol .hljs-string,.diff .hljs-change,.hljs-special,.hljs-attr_selector,.hljs-subst,.hljs-cdata,.css .hljs-pseudo,.hljs-header { color: #cb4b16 }
.hljs-deletion,.hljs-important { color: #dc322f }
.hljs-link_label { color: #6c71c4 }
.tex .hljs-formula { background: #eee8d5 }