当前位置: 首页 > news >正文

嵌入式面试问题:STM32中指针和数组的本质区别是什么,常用数组存储什么数据?

STM32中指针和数组的本质区别

核心本质区别

1.定义与内存分配

// 数组 - 静态分配,大小固定 uint8_t array[100]; // 编译器分配100字节连续内存 // array本身是内存地址的标识符,不是变量 // 指针 - 动态或静态,大小可变 uint8_t *ptr; // 只分配4/8字节存储地址 ptr = array; // 指向已存在内存 ptr = (uint8_t*)malloc(100); // 动态分配

2.类型特性

// sizeof行为不同 sizeof(array); // 返回整个数组大小 (100*sizeof(type)) sizeof(ptr); // 返回指针本身大小 (4或8字节) // 地址操作不同 &array; // 得到数组首地址,类型是uint8_t(*)[100] &ptr; // 得到指针变量的地址 // 自增操作不同 ptr++; // 指针移动sizeof(uint8_t)字节 array++; // 错误!数组名不是左值

3.内存布局差异

数组: ┌─────────────────┐ │ 数据直接存储 │ ← array标识符直接引用此内存 │ (连续100字节) │ └─────────────────┘ 指针: ┌─────────┐ ┌─────────────────┐ │ 地址值 │ → │ 实际数据 │ │ (4字节) │ │ (动态分配内存) │ └─────────┘ └─────────────────┘

STM32中常用数组存储的数据类型

1.外设数据缓冲区

// DMA传输缓冲区 uint8_t uart_rx_buffer[256]; // UART接收缓冲区 uint32_t adc_buffer[128]; // ADC采样数据 // 通信协议数据 uint8_t can_frame[8]; // CAN报文数据场 uint8_t spi_tx_data[64]; // SPI发送数据

2.信号处理数据

// 传感器数据缓存 int16_t imu_raw[6]; // 六轴IMU原始数据 float temperature_history[60]; // 温度历史记录 // 数字信号处理 float fft_input[1024]; // FFT输入数据 int32_t fir_filter_buffer[32]; // FIR滤波器缓存

3.系统状态与配置

// 系统状态数组 uint32_t task_stack[128]; // 任务堆栈空间 system_state_t states[10]; // 状态机状态 // 配置参数表 const uint32_t pwm_lookup[256] = { /* PWM占空比表 */ }; const float calibration_table[20]; // 校准系数

4.显示与界面数据

// 显示缓冲区 uint16_t lcd_frame_buffer[320*240]; // LCD显存 uint8_t led_matrix[8][8]; // LED点阵数据 // 字符与图形 const char menu_items[5][20]; // 菜单项文本 uint8_t bitmap_data[1024]; // 位图数据

5.实时控制系统数据

// 控制算法数据 float pid_error[3]; // PID误差记录 motor_position_t trajectory[100]; // 运动轨迹 // 采样与滤波 uint32_t encoder_counts[4]; // 编码器计数值 float current_samples[100]; // 电流采样

STM32中的特殊考虑

1.内存分段放置

// 指定数组存放位置 (链接脚本中定义) uint8_t __attribute__((section(".ccmram"))) fast_buffer[512]; // CCM RAM const uint8_t __attribute__((section(".rodata"))) lookup_table[256]; // 只读段

2.对齐要求

// DMA通常需要字对齐 uint32_t __attribute__((aligned(4))) dma_buffer[128]; // 缓存行对齐优化性能 uint8_t __attribute__((aligned(32))) cache_aligned_data[1024];

3.静态分配优势

无动态内存管理开销 - 适合实时系统
编译时确定大小 - 可预测的内存使用
更少的运行时错误 - 无内存泄漏/碎片化

选择建议

使用数组的场景:

  • 数据大小在编译时已知且固定

  • 需要频繁访问的缓冲区

  • 实时性要求高的中断服务程序

  • 避免动态内存管理的系统

使用指针的场景:

  • 数据大小在运行时确定

  • 需要灵活的内存管理

  • 传递数据到不同函数(避免大数组拷贝)

  • 动态数据结构(链表、树等)

STM32最佳实践:

// 优先使用静态数组 #define BUFFER_SIZE 256 static uint8_t local_buffer[BUFFER_SIZE]; // 文件内使用 // 需要传递时使用指针参数 void process_data(uint8_t *data, uint32_t length); // 大数组放置在合适的内存区域 __attribute__((section(".dtcm"))) uint32_t critical_buffer[1024]; // TCM内存

性能影响

  • 数组访问:编译时可计算偏移,通常更快

  • 指针访问:需要间接寻址,可能有额外开销

  • 缓存友好性:数组的连续内存更利于缓存预取

在STM32嵌入式开发中,由于内存有限且实时性要求高,大多数情况下推荐使用静态数组,除非确实需要动态内存分配。

http://www.jsqmd.com/news/92375/

相关文章:

  • 家长用华为,孩子用iPhone,怎样限制玩手机?
  • 阻抗控制未来展望:当前挑战和创新解决方案
  • 嵌入式面试问题:typedef在配置STM32寄存器中最常见的用途是什么?如何使用的?一个文章教会你如何封装函数
  • 自动控制原理中,什么样的动态过程是好的?
  • 计算机毕业设计|基于springboot + vue旅游信息推荐系统(源码+数据库+文档)
  • 计算机毕业设计|基于springboot + vue学生成绩管理系统(源码+数据库+文档)
  • Spring Boot4.0整合RabbitMQ死信队列详解
  • 【自动控制】自动控制原理中,如何用伯德图判定系统的性能?
  • 4.1 AI赋能代码研究:快速解构大型开源项目
  • FineReport 模拟题5 部门分级
  • 4.2 AI辅助技术文档撰写:将代码理解转化为专业文档
  • Qt中QSharedMemory析构处理指南
  • 2025广州比较好的留学中介有哪些 - 留学品牌推荐官
  • 家长如何一键管控孩子电脑时长?定时锁屏软件绿色版无需安装真能远程锁机吗
  • LobeChat角色预设功能实测:快速切换AI身份的便捷之道
  • 2025广州出国留学机构排名哪家口碑好一点 - 留学品牌推荐官
  • 2025广州出国留学中介机构前十排名有哪些 - 留学品牌推荐官
  • Higress云原生网关架构设计与生产环境部署实战
  • 上市公司元宇宙技术专利数据说明(1990—2025)
  • 从私人仓库到开源星系:技术传承的两种模式与工程师的职业跃迁
  • 软考-系统集成项目管理工程师案例简答题
  • 云贝餐饮V3全开源源码发布 支持独立连锁 全端Vue工程文件含全部插件
  • LobeChat能否支持HTTPS加密访问?SSL证书配置教程
  • 高并发系统性能测试中的用户数测算体系研究
  • 正则表达式的基础要点
  • 友达 G170ETN02.1 工业液晶显示屏:17.0 英寸超宽温高亮度场景的显示驱动技术解析
  • JVM内存模型详解
  • 3.2 AI Agent工作原理解析:任务分解与智能执行
  • 友达 G170ETN02.0 工业液晶显示屏:17.0 英寸超宽温高色域场景的显示驱动技术解析
  • 【软件工程与应用】平移置换搬迁系统设计与实现