0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

栈是一种LIFO后入先出的什么数据结构?

lhl545545 来源:玩转单片机 作者:玩转单片机 2020-06-09 09:26 次阅读

没有比这个更直观的啦,栈是一种受限的数据结构模型,其数据总是只能在顶部追加,利用一个指针进行索引,顶端叫栈顶,相对的一端底部称为栈底。栈是一种LIFO后入先出的数据结构。

栈就两种操作:

PUSH,压栈,向栈内加入数据,

POP,出栈

再进一步探讨:

首先将栈与堆分清,从看到这篇文章开始,我建议你不要把堆和栈连在一起叫,栈是栈,堆是堆,这是两回事,别混为一谈!(堆本文不深入讨论)

从C/C++编程语言的角度来看:

相同点:都是一片内存区,在链接时指定栈区/堆区的位置以及大小。

不同点:

栈:由编译器分配,存放函数的参数值,局部变量,寄存器组(不同的单片机/处理器各有不同)、函数调用参数传递、中断异常产生时须保存处理器状态的寄存器值等

堆:由程序员分配释放,对于C而言,malloc、realloc/free进行分配/释放,对C++而言,由new/delete分配/释放。

为啥用

栈这个数据模型的应用价值是什么呢?先来看一下单片机内部的可能有哪些栈应用?以STM32为例,参考IAR C/C++ Development Guide,P207

处理器模式建议段名描述

SupervisorSVC_STACK操作系统

IRQIRQ_STACK通用(IRQ)中断处理程序的堆栈。

FIQFIQ_STACK用于高速(FIQ)中断处理程序的堆栈。

UndefinedUND_STACK堆栈用于未定义的指令中断。支持硬件协处理器和指令集扩展的软件仿真

AbortABT_STACK用于指令获取和数据访问存储器中止中断处理程序的堆栈。

如果使用RTOS还有任务栈,如果是Linux,其内核线程同样也需要栈的支持,等等这一切的一切栈,其本质上都是利用了栈数据模型的LIFO后入先出的特性,一个典型应用场景就是比如做一件事情做到一半而要转而去做另外一件事,对于芯片编程而言,就需要将当前的工作做个暂存,等另外一件事情做完了,再接着回来继续做。那么怎么做到呢,以一个中断处理为例,要记住当前的工作态有哪些信息需要暂存呢?PC指针,局部变量等就被压入栈,再将中断服务程序地址导入PC指针,进而去执行中断服务程序,待中断处理完毕,在将栈里的内容按照后入先出弹出到对应的寄存器就恢复了原程序的现场,进而继续执行。

怎么用

栈在哪里定义大小,定多大合适?这可能很多刚接触单片机开发的同学不是太清楚,下面就将比较常见的IAR开发环境为例如何定义栈定义栈大小的地方说明一下,这里以IAR8.4.1为例,有两种方式可以进行栈大小设置。

IDE设置

为了更加清楚明了,制作了一个GIF操作展示视频,在stack/heap中就可以设置了,其中stack用于设置栈区大小,heap用于设置堆大小。

这个demo中设置了其栈的大小为0x200,堆的大小为0x400,全编译后,检查map文件就印证了栈/堆的大小如预期所修改。

链接配置文件

其实对于比较熟悉的开发人员,上一种方式并非推荐用法。用链接配置文件将具有更好的灵活性,比如可以指定一个段的对齐方式,存储位置,某个符号的存储位置等等。这里同样为了直观也做了一个GIF动画,介绍如何通过链接文件进行栈/堆的大小配置。

其最终的效果也一样如预期将栈区的大小设置好了。

栈溢出

这里为了比较容易的展示栈溢出的问题,在main函数利用递归方法计算阶乘,代码如下:

#include 《stdio.h》

#include “main.h”

static uint32_t spSatte[200];

static uint32_t spIndex = 0;

/*为什么要用浮点数,因为数据非常大整型很快就会溢出*/

float factorial(uint32_t n)

uint32_t sp = __get_MSP

/*记录栈指针的变化情况*/

spSatte[spIndex++] = sp;

if(n==0 || n==1)

return 1;

else

return (float)n*factorial(n-1);

int main(void)

float x = 0;

uint32_t n = 20;

printf(“stack test:

x = factorial(n);

/*打印栈指针变化情况*/

for(int i = 0;i《spIndex;i++)

printf(“MSP-》%08X

spSatte[i]);

/*打印阶乘结果*/

printf(“factorial(%d)=%f

n,x);

while (1)

为方便观察,将stm32f407xx_flash.icf 将栈改为256字节

/*stm32f407xx_flash.icf 将栈改为256字节*/

define symbol __ICFEDIT_size_cstack__ = 0x200;

define symbol __ICFEDIT_size_heap__ = 0x200;

全编译后通过map文件来看下栈/堆的分配情况:

“P2”, part 3 of 3: 0x400

CSTACK 0x2000‘05d8 0x200 《Block》

CSTACK uninit 0x2000’05d8 0x200 《Block tail》

HEAP 0x2000‘07d8 0x200 《Block》

HEAP uninit 0x2000’07d8 0x200 《Block tail》

- 0x2000‘09d8 0x400

直观些,翻译成下图,CSTACK段分配在0x2000 0280-0x2000 0480,堆分配在0x2000 0480-0x2000 0680。

栈是一种LIFO后入先出的什么数据结构?

图为什么没有将0x2000 07D8画在栈区呢?通过调试发现,这个字空间没有用做栈的实际存储。将工程设置成simulation模式,debug进入main.o勾选掉,我们来计算20的阶乘,来具体看一下:

栈是一种LIFO后入先出的什么数据结构?

对这个动图解读一下:

进入复位是,SP_main为0x200007D8,指向栈底,为空栈。那么这是怎么实现的呢?

__vector_table ;向量表

DCD sfe(CSTACK) ;这条命令会将程序的CSTACK起始地址装载给SP_main

DCD Reset_Handler ; Reset Handler复位向量

前面说0x200007D8并没有用到,怎么证明呢,在函数进入mian时,第一次压栈的情况如下:

栈是一种LIFO后入先出的什么数据结构?

可见STM32栈的增长方向是向下增长的,也即顶在小地址端一侧

栈存储元素是四字节对齐的,因为STM32的字长是字节,如果深入想想,如果不是司字节对齐会怎么样?留给感兴趣的思考一下。

0x200007D8--0x200007DB 这个字存储单元并不是栈的有效存储空间。

栈的变化情况:

stack test:

MSP-》200007A8

MSP-》20000790

MSP-》20000778

MSP-》20000760

MSP-》20000748

MSP-》20000730

MSP-》20000718

MSP-》20000700

MSP-》200006E8

MSP-》200006D0

MSP-》200006B8

MSP-》200006A0

MSP-》20000688

MSP-》20000670

MSP-》20000658

MSP-》20000640

MSP-》20000628

MSP-》20000610

MSP-》200005F8

MSP-》200005E0

factorial(20)=2432902023163674771.785700 /*结算结果与用计算器一致*/

每调用一次阶乘函数,栈就压入4个字,由上面还可以看到第20次进入时,栈指针为0x200005E0,如果再压入4个字栈指针会变成0x200005C8,是这样吗,结果还对吗?将n改为21编译运行,来看一看:

看到了吧,惊喜来了,栈溢出了,程序已经不听话了,完全不知道在干嘛了。所以栈溢出的后果是极端危险的,完全无法预期,程序会带来什么后果。

总结一下

栈是一种LIFO后入先出的数据结构模型,是C/C++程序运行时基础,没这个栈,C/C++玩不转

栈在嵌入式编程领域随处可见,比如C栈,中断栈、异常栈、任务栈等等,但其基本工作原理都一样。支持两种基本数据操作:压栈、出栈。

栈溢出程序的结果无法预期,所以合理的设置栈区大小是个永恒的话题,过大则浪费内存,过小则程序会飞。

嵌入式编程递归函数要慎用,个人建议不用。比如IEC61508 功能安全标准中强行规定不可使用递归函数。

STM32中__get_MSP可以得到当前栈指针的值,据此可以做一定程度的栈溢出保护措施。防止程序跑飞。

通过上面递归调用测试,还可以得到一个启示,嵌入式编程函数嵌套的层级不宜过深,过深则需要相对较大的栈开销。
责任编辑:pj

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 编译器
    +关注

    关注

    1

    文章

    1623

    浏览量

    49104
  • 数据结构
    +关注

    关注

    3

    文章

    573

    浏览量

    40121
收藏 人收藏

    评论

    相关推荐

    hart协议的协议结构分析

    的智能化水平,同时保持与现有模拟系统的兼容性。 2. HART协议概述 HART协议可以分为几个层次,每个层次负责不同的功能: 2.1 物理层(Physical Layer) 物理层负责在物理媒介上传输数据。HART协议使用
    的头像 发表于 12-02 09:43 186次阅读

    DDC264配置寄存器数据写入和320 DCLK时钟脉冲的回读数据结构是什么?

    配置寄存器数据写入和320 DCLK时钟脉冲的回读数据结构是什么? 根据注和表9,16位配置寄存器数据,4位修订ID, 300位校验模式,怎么可能有1024 TOTAL READ
    发表于 11-19 07:58

    视觉软件HALCON的数据结构

    在研究机器视觉算法之前,我们需要先了解机器视觉应用中涉及的基本数据结构。Halcon数据结构主要有图像参数和控制参数两类参数。图像参数包括:image、region、XLD,控制参数包括:string、integer、real、handle、tuple数组等。
    的头像 发表于 11-14 10:20 333次阅读
    视觉软件HALCON的<b class='flag-5'>数据结构</b>

    嵌入式环形队列与消息队列的实现原理

    嵌入式环形队列,也称为环形缓冲区或循环队列,是一种先进先出(FIFO)的数据结构,用于在固定大小的存储区域中高效地存储和访问数据。其主要特点包括固定大小的数组和两个指针(头指针和尾指针
    的头像 发表于 09-02 15:29 467次阅读

    嵌入式常用数据结构有哪些

    在嵌入式编程中,数据结构的选择和使用对于程序的性能、内存管理以及开发效率都具有重要影响。嵌入式系统由于资源受限(如处理器速度、内存大小等),因此对数据结构的选择和使用尤为关键。以下是嵌入式编程中常用的几种数据结构,结合具体特点和
    的头像 发表于 09-02 15:25 456次阅读

    请问esp event的时序逻辑是怎样的?

    的.在个模块里, 如果notify A再notify B, 回调的执行顺序是先进入cb_B, 再进入cb_A, 看上去是先出
    发表于 06-07 06:57

    OpenHarmony语言基础类库【@ohos.util.Queue (线性容器Queue)】

    Queue的特点是先进先出,在尾部增加元素,在头部删除元素。根据循环队列的数据结构实现。
    的头像 发表于 04-27 21:20 326次阅读
    OpenHarmony语言基础类库【@ohos.util.Queue (线性容器Queue)】

    探索编程世界的七大数据结构

    结构就像是颗倒挂的小树,有根、有枝、有叶。它是一种非线性的数据结构,以层级的方式存储数据,顶部是根节点,底部是叶节点。
    的头像 发表于 04-16 12:04 384次阅读

    TASKING编译器是否可以将数据结构设置为 \"打包\"?

    TASKING 编译器是否可以将数据结构设置为 \"打包\"? GCC 很早以前就提供了这种可能性,可以将__attribute__((packed))与对齐指令结合使用。 对于
    发表于 03-05 06:00

    矢量与栅格数据结构各有什么特征

    数据结构是使用点、线和面等基本几何图形来描述和表示地理对象的一种方法。它们由离散的几何对象和与之相关的属性数据组成。矢量数据中的点表示个特
    的头像 发表于 02-25 15:06 2522次阅读

    TC397B CAN00节点设置的接收模式是先先出,并且设置了新消息中断,为什么没有效果?

    CAN00节点设置的接收模式是先先出,并且设置了新消息中断,但是没有效果
    发表于 02-04 06:48

    堆和的区别和使用注意事项

    堆和是在计算机科学中广泛使用的两种数据结构,它们具有不同的用途和特点。堆和的区别涉及到内存分配、访问方式、数据存储等方面。在使用堆和
    的头像 发表于 01-18 17:24 2174次阅读

    区块链是什么样的数据结构组织

    区块链是一种特殊的数据结构,它以分布式、去中心化的方式组织和存储数据。区块链的核心原理是将数据分布在网络的各个节点上,通过密码学算法保证数据
    的头像 发表于 01-11 10:57 2199次阅读

    结构体与指针的关系

    在C语言中,结构体(Struct)是一种用户自定义的数据类型,它允许您将不同类型的数据项组合在起,以便形成
    的头像 发表于 01-11 08:00 974次阅读
    <b class='flag-5'>结构</b>体与指针的关系

    C语言数据结构之跳表详解

    大家好,今天分享篇C语言数据结构相关的文章--跳表。
    的头像 发表于 12-29 09:32 822次阅读
    C语言<b class='flag-5'>数据结构</b>之跳表详解