本节主要说程序中的栈函数栈的关系以及栈和递归算法的关系。

一、函数调用时的栈

1.程序调用时的栈是也就是平时所说的函数栈是数据结构的一种应用,函数调用栈一般是从搞地质向低地址增长的,栈顶为内存的低地址,栈底为内存的高地址。函数调用栈中存储的是数据的活动记录。活动记录是函数一些信息。如下如所所示:

2.假如有如下程序:

#include <stdio.h>
#include <stdlib.h> void fun(char* a)
{
char* b;
strcpy(b,a);
}
int main()
{
char a = '1';
fun(&a);
return 0;
}

那么在函数调用的过程中,函数栈的活动记录如下所示:

3.程序中的栈可以看做是顺序栈的一个应用,栈中保存了一个函数调用所需要的维护信息,包括函数参数,函数返回值地址,局部变量,函数调用上下文等。

4.具体的函数中的栈的实现请看另一篇博文:http://blog.csdn.net/cheyuxuan/article/details/9722463

二、程序中的栈

1.前一阶段在某实训机构做了一个所谓的项目时候,里面使用到了一个冒泡排序,但是排序的数据量很大,大约有6万个,程序的复杂度是O(n2),在程序运行一会后就会自动跳出,后来经过问大牛才知道这是因为栈溢出的原因。
2.程序在运行过程中,函数都是要被压入栈中的, 所以我们不断的调用函数,那么就会导致栈空间溢出,程序直接死掉。

3.造成栈溢出的原因通常有如下两点:①递归过深②局部数组过大,因为每一次递归都要调用函数,都要进行压栈的操作,所以这样会导致栈溢出,同时,如果我们定义的局部数组过大,数组也是在栈上分配空间的,那么也会造成栈上的空间不足,造成栈溢出。

三、程序中的栈的应用----递归调用

1.递归的算法是任何一门语言的基本功,但是理解起来其实也比较困难,这节先不说递归调用算法的实现,先来说一下递归和栈的关系。

2.例程:

#include <stdio.h>

void reverse(char* s)
{
if( (s != NULL) && (*s != '\0') )
{
reverse(s + 1); printf("%c", *s);
}
} int main()
{
reverse("12345"); printf("\n"); return 0;
}

程序打印的结果如下:

从程序大打印结果我们可以看出,我们实现了字符串"12345",最关键的是我们并没有使用我们惯用的for和count的方法。

解析:

我们从主函数调用reverse函数,向reverse函数传递的参数是字符串"12345"的首地址,然后进入子函数reverse,进行两个安全性的检测以后,我们再次调用reverse这个函数,这个时候向这个函数传递的是字符串"12345"的第二个元素的地址,依次类推,最后我们向reverse函数传递的是最后一个字符串的地址,这个时候程序中的栈被经历了5次reverse函数的压入,在第五次运行结束以后,栈中的值依次弹出,首先弹出的是第五个元素,然后以此类推。最后打印出我们上面的打印结果。

具体过程如下图所示:

程序在递归调用期间不断的向栈中压入每一次函数调用的记录,在递归调用结束后再从栈栈弹出每一次的记录,所以就打印出了我们看到的值。

四、总结

1.程序栈空间在本质上是一种顺序栈。

2.程序栈空间的访问是通过函数调用执行的。

3.程序栈空间仍然遵守后进先出的原则。

5.问题:①某实训机构的老师说程序的栈空间时从两侧增长的,我没有想懂,首先那那违背了栈的存储结构,其次同学早上和我说实际上栈的调度是CPU在控制的。

②程序中的栈空间,数据结构中的栈,还有真正的汇编硬件中的栈那个关系还需要深度学习,现在心里大概懂一点,但是无法形成一个整体的脉络。

C语言数据结构----栈与递归的更多相关文章

  1. C语言数据结构-栈的实现-初始化、销毁、长度、取栈顶元素、查找、入栈、出栈、显示操作

    1.数据结构-栈的实现-C语言 #define MAXSIZE 100 //栈的存储结构 typedef struct { int* base; //栈底指针 int* top; //栈顶指针 int ...

  2. C语言数据结构-栈

    一.栈的定义 栈(statck)这种数据结构在计算机中是相当出名的.栈中的数据是先进后出的(First In Last Out, FILO).栈只有一个出口,允许新增元素(只能在栈顶上增加). 移出元 ...

  3. C语言数据结构----栈的应用(程序的符号匹配检测)

    本节主要讲利用栈来实现一个程序中的成对出现的符号的检测,完成一个类似编译器的符号检测的功能,采用的是链式栈. 一.问题的提出以及解决方法 1.假定有下面一段程序: #include <stdio ...

  4. C语言数据结构----栈的定义及实现

    本节主要说的是数据结构中的栈的基本定义和实现的方式,其中实现的方式采用的是复用顺序表和单向链表的方式. 一.栈的基本定义 1.栈是一种特殊的线性表,只能从固定的方向进出,而且栈进出的基本原则是:先进栈 ...

  5. 数据结构——栈与递归(recursion)

    /* recursion.c */ /* 递归 */ #include <stdio.h> void interface(void); /* 斐波那契数列以及阶乘函数声明 */ long ...

  6. C语言数据结构栈

    #include<stdio.h>#include<stdlib.h>typedef struct Node{ int data; struct Node* pnext;}no ...

  7. 数据结构——栈(C语言实现)

    #include <stdio.h> #include <stdlib.h> #include<string.h> #include<malloc.h> ...

  8. C语言数据结构之栈:中缀表达式的计算

    *注:本人技术不咋的,就是拿代码出来和大家看看,代码漏洞百出,完全没有优化,主要看气质,是吧 学了数据结构——栈,当然少不了习题.习题中最难的也是最有意思的就是这个中缀表达式的计算了(可以算+-*/和 ...

  9. (js描述的)数据结构[栈结构](2)

    (js描述的)数据结构[栈结构](2) 一.什么是栈结构 1.一种受限制的线性结构,这种结构可以基于数组来实现. 2.可以抽象成一个容器,上面的是栈顶,底下的是栈底.所以仅允许对栈顶进行操作, 二.栈 ...

随机推荐

  1. 不要伤害指针(5)--void和void指针详解

    原文转载地址:http://blog.csdn.net/sunchaoenter/article/details/6587426 增加自己的想法,作为笔记. 1.概述 许多初学者对C/C++语言中的v ...

  2. day9_python学习笔记_chapter12_模块

    1. 名称空间加载顺序: 首先加载内建名称空间,他由__builtin模块中的名字构成.然后加载执行模块的全局名称空间,他会在模块开始执行后变为活动名称空间.如 果在执行期间调用了一个函数,那么将创建 ...

  3. cursor的形状

    Example:CSS鼠标手型效果 <a href="#" style="cursor:hand">CSS鼠标手型效果</a> Exam ...

  4. PHP中mktime() 函数对于日期运算和验证

    mktime() 函数对于日期运算和验证非常有用.它可以自动校正越界的输入: // 语法:mktime(hour,minute,second,month,day,year) echo(date('Y- ...

  5. Javascript面向对象之创建对象

    面向对象的语言具有一个共同的标志,那就是具有“类”的概念,但是在javascript中没有类的概念,在js中将对象定义为“无序属性的集合,其属性可以包含基本值,对象或者函数”,即其将对象看作是一组名值 ...

  6. Oracle EBS-SQL (BOM-4):检查期间新增编码总数.sql

    selectFU.description                                    创建者,msi.CREATION_DATE                        ...

  7. DbConnectionFactory 数据库连接

    /** * */package com.sprucetec.dbatch.tmsfee;import java.io.Serializable;import java.sql.Connection;i ...

  8. thenjs的应用

    <!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. USB OTG简单介绍

    1 引言 随着USB2.0版本号的公布,USB越来越流行,已经成为一种标准接口.如今,USB支持三种传输速率:低速(1.5Mb/s).全速(12Mb/s)和快速(480Mb/s),四种传输类型:块传输 ...

  10. Struts2.x教程(一) Struts2介绍

    一.Struts2是什么 Struts2是在WebWork2基础发展而来的.和Struts1一样, Struts2也是基于MVC的web层框架. 那么既然有了Struts1,为何还要Struts2? ...