C语言变量定义与单片机数据储存方式

2024-05-07 18:32

本文主要是介绍C语言变量定义与单片机数据储存方式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

说明:文章来源

EDN电子技术设计:嵌入式程序开发需要知道的存储器知识

MCU 中常使用的存储器类型有:FLASH、RAM、ROM(包括EEPROM)
在软件角度来看,程序和数据的存储分为以下几个部分
这里写图片描述

代码段和常量段都可以用于保存常量数据,其主要区别是,如果常量可以作为汇编指令的一个操作数,则该常量被编译进代码段。如果不能用一个汇编操作数表示,则存于常量段。如 “uchar a=0x05;” 中的 “0x05” 将被编译成代码 “mov #0x05, a”;如果是 “uchar a[]={0x05, 0x06}” 则 “0x05,0x06” 被放置于常量段, 在初始化 a[] 的时候会有一段汇编指令用于将常量段中的内容拷贝到 a[] 中。

软件存储区与硬件存储器类型是怎么对应的呢?一般来讲如下:
这里写图片描述

1.MCU 中的 ROM 通常用于存储制造商信息、控制器型号等信息;
2.对于 x86 体系结构的系统,因为没有 Flash 类型的存储器,所以,所有的软件存储区最终都加载到内存中,但是其内存是分段的,用户对不同内存段的访问权限不同,其代码段和常量段不可以被用户修改,如果意外修改则抛出段错误异常。

知道了存储器类型和各存储区的划分之后,让我们来看以下三组程序:

static void ProcStr(void)
{uchar Str[] = {"12345"};
}

这段程序中,Str[] 是一个局部数组,其大小为 6,占用的堆栈空间是 2 个字符;”12345” 是常量,被存储在常量段;Str[] 的初始化过程,相当于从常量区拷贝 6 个字符的数据到栈中,这 6 个字符是”12345\0”。

static void ProcStr(void)
{uchar Str[] = "12345";
}

这段程序中,Str[] 是一个局部数组,其大小为 6,占用的堆栈空间是 2 个字符;”12345” 是常量,被存储在常量段;Str[] 的初始化过程,相当于从常量区拷贝 6 个字符的数据到栈中,这 6 个字符是 “12345\0”。

static void ProcStr(void)
{const uchar* Str = "12345";
}

这段程序中没有数组,唯一的 Str 是一个局部指针,其大小为 4(在 32 位系统中),因此这段程序只占用 4(在 32 位系统中) 个字符的堆栈空间;”12345”是常量,被存储在常量段;Str 的初始化过程,是将指针 Str 初始化为常量”12345”的地址,后续程序通过指针 Str 直接访问常量段,无需内存拷贝过程。


从以上分析可以看出,前两种方法是一样的,都需要为局部数据分配存储空间,并将静态存储区的数据拷贝过来,而最后一种方法是通过指针直接访问静态数据而无需拷贝。如果字符串长度大于系统中指针的长度,第三种方法将在时间和空间上大大优于前两种方法(第三种方法极大的节省了堆栈空间,并减少了拷贝数据所用的时间)。

但是,对于 MCU 来说,并不总是第三种方法好,原因在于第三种方法是直接访问常量段,由上面的表可知,对于将常量存储于 Flash 的 MCU 来说,访问常量段要比访问 RAM 慢得多。因此,如果接下来要频繁访问这个字符串,那么,采用前两种方法在速度上将会更优一些,理由是前两种方法只需要访问一次 Flash,而第三种方法则每次都需要访问 Flash。

当然,如果在接下来的程序中,需要修改字符串 Str 中的内容,那就只能采用前两种方法,第三种方法将会提示错误。

特殊说明:今天讲的一些内容跟编译器的特性相关,不同编译器,甚至相同编译器的不同版本间存在一定差异。

这篇关于C语言变量定义与单片机数据储存方式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/968055

相关文章

解决mysql插入数据锁等待超时报错:Lock wait timeout exceeded;try restarting transaction

《解决mysql插入数据锁等待超时报错:Lockwaittimeoutexceeded;tryrestartingtransaction》:本文主要介绍解决mysql插入数据锁等待超时报... 目录报错信息解决办法1、数据库中执行如下sql2、再到 INNODB_TRX 事务表中查看总结报错信息Lock

MySQL 添加索引5种方式示例详解(实用sql代码)

《MySQL添加索引5种方式示例详解(实用sql代码)》在MySQL数据库中添加索引可以帮助提高查询性能,尤其是在数据量大的表中,下面给大家分享MySQL添加索引5种方式示例详解(实用sql代码),... 在mysql数据库中添加索引可以帮助提高查询性能,尤其是在数据量大的表中。索引可以在创建表时定义,也可

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

Linux lvm实例之如何创建一个专用于MySQL数据存储的LVM卷组

《Linuxlvm实例之如何创建一个专用于MySQL数据存储的LVM卷组》:本文主要介绍使用Linux创建一个专用于MySQL数据存储的LVM卷组的实例,具有很好的参考价值,希望对大家有所帮助,... 目录在Centos 7上创建卷China编程组并配置mysql数据目录1. 检查现有磁盘2. 创建物理卷3. 创

Nacos日志与Raft的数据清理指南

《Nacos日志与Raft的数据清理指南》随着运行时间的增长,Nacos的日志文件(logs/)和Raft持久化数据(data/protocol/raft/)可能会占用大量磁盘空间,影响系统稳定性,本... 目录引言1. Nacos 日志文件(logs/ 目录)清理1.1 日志文件的作用1.2 是否可以删除

ShardingSphere之读写分离方式

《ShardingSphere之读写分离方式》:本文主要介绍ShardingSphere之读写分离方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录ShardingSphere-读写分离读写分离mysql主从集群创建 user 表主节点执行见表语句项目代码读写分

使用Python获取JS加载的数据的多种实现方法

《使用Python获取JS加载的数据的多种实现方法》在当今的互联网时代,网页数据的动态加载已经成为一种常见的技术手段,许多现代网站通过JavaScript(JS)动态加载内容,这使得传统的静态网页爬取... 目录引言一、动态 网页与js加载数据的原理二、python爬取JS加载数据的方法(一)分析网络请求1

golang float和科学计数法转字符串的实现方式

《golangfloat和科学计数法转字符串的实现方式》:本文主要介绍golangfloat和科学计数法转字符串的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望... 目录golang float和科学计数法转字符串需要对float转字符串做处理总结golang float

linux lvm快照的正确mount挂载实现方式

《linuxlvm快照的正确mount挂载实现方式》:本文主要介绍linuxlvm快照的正确mount挂载实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux lvm快照的正确mount挂载1. 检查快照是否正确创建www.chinasem.cn2.

SpringBoot项目Web拦截器使用的多种方式

《SpringBoot项目Web拦截器使用的多种方式》在SpringBoot应用中,Web拦截器(Interceptor)是一种用于在请求处理的不同阶段执行自定义逻辑的机制,下面给大家介绍Sprin... 目录一、实现 HandlerInterceptor 接口1、创建HandlerInterceptor实