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

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

3天内不再提示

嵌入式里堆栈原理是怎么样的如何进行纯C实现

Wildesbeast 来源:与非网 作者:与非网 2020-02-06 16:23 次阅读

栈这种结构在嵌入式里其实是非常常用的,比如函数调用与返回就是典型的栈应用,虽然很多时候栈都是CPU系统在自动管理,我们只需要在链接文件里分配栈大小以及栈存放位置,但稍微了解一下栈的原理会更加利于我们去理解嵌入式代码执行机制,以及帮助我们进一步去调试。

1. 何为堆栈?

堆 HEAP与栈 STACK 是两个不同概念,其本质上都是一种数据结构。

栈是一种按数据项排列的数据结构,只能在一端(栈顶 top)对数据项进行插入和删除,其符合后进先出(Last-In / First-Out)原则。栈(os)一般是由编译器自动分配释放,其使用的是一级缓存。

堆也是一种分配方式类似于链表的数据结构,其可以在任意位置对数据项进行操作。堆(os)一般由程序员手动分配释放,其使用的是二级缓存。

在嵌入式世界里,堆栈一般指的仅是栈。

2. 作用与意义

MCU 中,栈这种结构一般被 cpu 和 os 所使用。

在 cpu 裸机中使用情况分两种:一、主动进行函数调用时,STACK 用以暂存下一条指令地址、函数参数、函数中定义的局部变量;二、硬中断来临时,暂存当前执行的现场数据(下一条指令地址、各种缓存数据),中断结束后,用以恢复。

在 os 中使用时,硬栈的使用同 cpu 裸机;但 os 一般会为每个任务额外分配一个软栈,在任务调度时,可用软中断打断当前正在执行的任务,栈则用以保存各自任务以恢复。

3. 软硬之分

硬件堆栈:是通过寄存器 SP 作为索引指针的地址,是调用了 BL 等函数调用指令后硬件自动填充的堆栈。

软件堆栈:是编译器为了处理一些参数传递而做的堆栈,会由编译器自动产生和处理,可以通过相应的编译选项对其进行编辑。

简单一点说,硬件堆栈主要做为地址堆栈用,而软件堆栈主要会被分配成数据堆栈。或看其栈顶指针是否和 CPU 具有特殊的关联,有关联者(如 SP)“硬”,而无关联者“软”。

4. 栈的纯 C 实现

基本的抽象数据类型(ADT)是编写 C 程序必要的过程,这类 ADT 有链表、堆栈、队列和树等,本节主要讲解下堆栈的几种实现方法以及他们的优缺点。

堆栈(stack)的显著特点是后进先出(Last-In First-Out, LIFO),其实现的方法有三种可选方案:静态数组、动态分配的数组、动态分配的链式结构。

静态数组:特点是要求结构的长度固定,而且长度在编译时候就得确定。其优点是结构简单,实现起来方便而不容易出错。而缺点就是不够灵活以及固定长度不容易控制,适用于知道明确长度的场合。

动态数组:特点是长度可以在运行时候才确定以及可以更改原来数组的长度。优点是灵活,缺点是由此会增加程序的复杂性。

链式结构:特点是无长度上线,需要的时候再申请分配内存空间,可最大程度上实现灵活性。缺点是链式结构的链接字段需要消耗一定的内存,在链式结构中访问一个特定元素的效率不如数组。

首先先确定一个堆栈接口的头文件,里面包含了各个方案下的函数原型,放在一起是为了实现程序的模块化以及便于修改。然后再接着分别介绍各个方案的具体实施方法。

堆栈接口 stack.h 文件代码:

4.1 静态数组

在静态数组堆栈中,STACK_SIZE 表示堆栈所能存储的元素的最大值,用 top_element 作为数组下标来表示堆栈里面的元素,当 top_element == -1 的时候表示堆栈为空;当 top_element == STACK_SIZE - 1 的时候表示堆栈为满。push 的时候 top_element 加 1,top_element == 0 时表示第一个堆栈元素;pop 的时候 top_element 减 1。

a_stack.c 源代码如下:

4.2 动态数组

头文件还是用 stack.h,改动的并不是很多,增加了 stack_size 变量取代 STACK_SIZE 来保存堆栈的长度,数组由一个指针来代替,在全局变量下缺省为 0。

create_stack 函数首先检查堆栈是否已经创建,然后才分配所需数量的内存并检查分配是否成功。destroy_stack 函数首先检查堆栈是否存在,已经释放内存之后把长度和指针变量重新设置为零。is_empty 和 is_full 函数中添加了一条断言,防止任何堆栈函数在堆栈被创建之前就被调用。

d_stack.c 源代码如下:

4.3 链式结构

由于只有堆栈顶部元素才可以被访问,因此适用单链表可以很好实现链式堆栈,而且无长度限制。把一个元素压入堆栈是通过在链表头部添加一个元素实现。弹出一个元素是通过删除链表头部第一个元素实现。由于没有长度限制,故不需要 create_stack 函数,需要 destroy_stack 进行释放内存以避免内存泄漏。

l_stack.c 源代码如下:

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

    关注

    4976

    文章

    18246

    浏览量

    287938
  • 寄存器
    +关注

    关注

    30

    文章

    5020

    浏览量

    117630
  • cpu
    cpu
    +关注

    关注

    68

    文章

    10412

    浏览量

    206467
收藏 人收藏

    评论

    相关推荐

    [分享]如何进嵌入式开发的世界?

    何进嵌入式开发的世界?我们认为,首先掌握C语言的指针和内存管理等高级技术,然后掌握一些硬件知识,最后可以通过最容易找到的嵌入式Linux进行
    发表于 05-22 11:03

    你是如何理解嵌入式C编程

    、新手常常问的一个问题是c语言和嵌入式c编程有什么区别?而嵌入式工程师一般都会告诉你其区别在于嵌入式C
    发表于 03-22 10:58

    你是如何理解嵌入式c编程的

    嵌入式开发感兴趣的童鞋不妨接着往下看。 一、新手常常问的一个问题是c语言和嵌入式c编程有什么区别?而嵌入式工程师一般都会告诉你其区别在于
    发表于 05-05 15:11

    何进行嵌入式实时PLC设计?

    基于 VxWorks 操作系统,探讨如何进行嵌入式实时PLC设计?利用VxWorks 的开放性、模块化和可扩展性的系统结构特性以及多线程/多任务的系统环境来达到高实时要求的PLC 控制,在保证实时性的同时,实现多点位、复杂功能的
    发表于 07-31 06:02

    何进行Ubuntu下Qt嵌入式交叉编译开发环境搭建?

    何进行Ubuntu下Qt嵌入式交叉编译开发环境搭建?
    发表于 09-11 02:42

    常用知识:嵌入式堆栈原理及其C实现

    常用知识:嵌入式堆栈原理及其C实现  大家好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给大
    发表于 04-13 17:20

    常用知识:嵌入式堆栈原理及其C实现

    常用知识:嵌入式堆栈原理及其C实现 大家好,我是正经搞技术的痞子。今天给大家讲的是
    发表于 04-17 10:19

    嵌入式学习手册——堆栈结构与编程

    可以在任意位置对数据项进行操作。堆(os)一般由程序员手动分配释放,其使用的是二级缓存。  在嵌入式世界堆栈一般指的仅是栈。2. 作用与意义在 MCU 中,栈这种结构一般被 cpu
    发表于 04-20 07:00

    软件和嵌入式软件区别

      软件和嵌入式软件区别为:  1、定义不同  ①嵌入式软件就是嵌入在硬件中的操作系统和开发工具软件,包括软件部分和硬件部分;  ②软件
    发表于 06-28 11:36

    嵌入式堆栈原理及其C实现方法

    嵌入式堆栈原理及其C实现  
    发表于 12-28 06:30

    嵌入式系统使用的存储器是如何进行划分的

    嵌入式最小硬件系统是由哪些部分组成的?嵌入式系统使用的存储器是如何进行划分的?可分为哪几类?
    发表于 10-22 07:18

    为何要进行嵌入式软件架构设计?如何设计?

    为何要进行嵌入式软件架构设计?如何进行嵌入式软件架构设计?
    发表于 11-01 06:31

    嵌入式C语言应用工程中堆栈的相关资料分享

    堆栈的定义和作用2. 嵌入式 C 语言应用工程的栈大小确定3. 嵌入式 C 语言应用工程的堆栈
    发表于 11-04 06:10

    嵌入式产品如何进行安全防护

    上期提到了嵌入式MCU破解技术,虽不全面,但足够起到警示作用。本期主要讲述嵌入式产品如何进行安全防护。 因为MCU端的程序很容易被获取到,所以MCU端的程序和数据都是不安全的。最直接的加密防护,就是
    发表于 11-04 07:33

    应用热更新如何进行嵌入式设备开发调试?

    应用热更新如何进行嵌入式设备开发调试?
    发表于 12-23 06:02