|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
C语言程序的动态内存分为栈内存区域和堆内存区域两种。栈内存是由编译器管理的,而堆内存是由程序调用具体的库函数管理的。我们今天分析下栈内存的概念。
, |3 q- v7 u% W& B5 F' M6 {! p% R; I: [; n" A0 ~
3 P; E. T) \7 S9 \1 @1 N- G栈内存的使用在很大程度上依赖于处理器的硬件机制。在处理器中,一般有一个寄存器来表示当前栈指针的位置,通常在内存中分配一块区域,这块内存的上界(高内存地址)和下界(低内存地址)之间是可用的栈内存区域。, m, _' E, |, D6 T3 l" h
, _- q! G4 S B1 I7 Z栈指针是一个指向栈区域内部的指针,也就是它的值是一个地址,这个地址位于栈区的下界和栈区的上界之间。栈指针把这个栈区域分为两个部分,一个是已经使用的区域,一个是没有使用的区域。
; v* T' `) L' C4 j" |' I& h3 n) a" f3 B) B
对于栈内存的增长方向有两种:一种是向上增长的,也就是低地址向高地址增长;另一个是向下增长的,高地址向低地址增长。在目前常见的体系结构和编译系统中,栈大多是向下增长的,我们也是看下这种常见的增长形式。在初始阶段,栈指针是指向栈区间的上界。随着栈使用量的增加,栈指针的值将向低地址移动,也就是在变小。' r' q' \9 e* B( Y! T# Z* d
* p6 R0 q; |/ Q8 t
栈内存在使用过程中有一个重要的特性是先入后出,也就是后入栈的内容将先出栈,而先入栈的后出栈。类似于一个口的瓶子,先进去的在底下,要想底下的出来就先把上面的先倒出来。栈内存的使用情况见下图:
* n I# Z! O: c# E* H
! z7 _# J1 q, w
入栈的过程和出栈的过程我们安全用图形来表示,更形象些吧~
6 T$ _: X8 d; A* | Y! t
5 s- c: A$ F, V5 ]/ v- X
, q% Z! }' e) r: G$ i在入栈的过程中,如果栈指针的变化超出栈内存的区域,将发生栈溢出。; \; M$ \, M; l- ` O9 V" B
. v) r6 N' s8 w2 X$ M( l从图中看出栈指针的功能是标识当前的栈位置。对栈内存处理中,每次能够获取的内容都是最后可放入栈内存的内容,而每次放入栈内存中的内容都将位于栈区域的最后。3 k3 |8 O5 @% g5 K4 q/ U
( M/ V& j' ?7 [2 B/ E
总的来说其实栈是一个先入后出的内存区域,栈指针是提供一种硬件的内存机制。
& W6 P8 S! F0 D. X) Z- E2 Y
$ |" C% o& z9 m" f# Q2 z+ \还有一个大家可能都没听说过,或者都没关注过的,我们来一起了解一下,就是满栈和空栈的概念,我们还是通过图来形容一下,这个是由处理器的体系结构决定的。与程序的编写没有关系,甚至编译器都不需要关注这个问题。无论在哪种情况下,栈指针都是已经使用的栈区域和未使用的栈区域的分界线。 z0 D T) \$ H3 D! h2 Z& }2 ]
8 c5 H/ I/ [2 S' F- l. t4 X! U
* W0 o9 u& L8 d$ `: K6 u/ a5 b' |3 u( D
在满栈的情况:栈指针当前的位置是已经使用的栈区域。
( M; T' p+ A! ~+ q; s" T7 B! B' V4 l* r# F0 c, e: e( K
在空栈的情况:栈指针当期的位置是没有使用的栈区域。 I1 n4 r! B. }4 R: P
9 V2 ?, ~6 e3 {$ t" L5 x这个仅供大家了解下就可以了,毕竟对于我们大多数人来说都是应用者,多了解点底层的总没错,但也不必太深挖。对于栈内存的概念我就分享到这里,其实这个对于写汇编的人来说就很有用处了,或者去多读一些汇编就很能清楚栈内存的妙用了。后续我分享堆内存的一些概念,话说知识是一点点积累的过程,有时候觉得前面有的知识点懵懵懂懂的突然连起来就又通透了。这就是坚持学习的作用,希望大家都能坚持多学,才能更会用。" Y4 o! }& u @; V* V) f1 x* Q
/ L" K: C b" T3 c& K4 b1 x+ w) F" i
5 @9 V* h2 l3 u * ^* G- ?/ t) y% q
' `, K* q1 |' l, O* L' R4 k( }
|
|