news 2026/5/12 17:49:53

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
嵌入式面试问题: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嵌入式开发中,由于内存有限且实时性要求高,大多数情况下推荐使用静态数组,除非确实需要动态内存分配。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/11 13:29:29

LobeChat角色预设功能实测:快速切换AI身份的便捷之道

LobeChat角色预设功能实测:快速切换AI身份的便捷之道 在如今这个大模型遍地开花的时代,几乎每个团队都在尝试搭建自己的AI对话系统。但很快就会遇到一个现实问题:同一个AI,怎么能既当严谨的数据分析师,又做幽默风趣的朋…

作者头像 李华
网站建设 2026/5/12 15:36:24

LobeChat能否支持HTTPS加密访问?SSL证书配置教程

LobeChat 能否支持 HTTPS 加密访问?SSL 证书配置实战指南 在 AI 应用加速落地的今天,越来越多开发者选择将 LobeChat 部署为私有化的智能对话门户。这款基于 Next.js 的开源聊天界面以优雅的设计、灵活的插件系统和对多模型的良好兼容性,成为…

作者头像 李华
网站建设 2026/5/8 9:59:21

深度学习优化策略:从理论到实践的完整指南

深度学习优化策略:从理论到实践的完整指南 【免费下载链接】nndl.github.io 《神经网络与深度学习》 邱锡鹏著 Neural Network and Deep Learning 项目地址: https://gitcode.com/GitHub_Trending/nn/nndl.github.io 在神经网络训练过程中,你是否…

作者头像 李华
网站建设 2026/5/10 14:53:32

Claude-Opus-4.5国内接入实战:从网络层到业务层的工程化落地

在Claude-Opus-4.5发布后,在复杂推理场景下的表现引起了开发界的广泛关注。但对于国内团队而言,如何将稳定集成到生产环境,面临着网络、支付与 SDK 兼容这“三座大山”。本文基于实际开发经验,拆解接入难点并分享了一套其高可用的…

作者头像 李华
网站建设 2026/5/10 19:22:08

5.2 MCP协议详解:核心机制与设计思想

5.2 MCP协议详解:核心机制与设计思想 在上一节课中,我们深入探讨了LLM的两个致命痛点:上下文窗口限制和知识滞后性。为了解决这些问题,业界提出了Model Context Protocol (MCP)这一创新性解决方案。本节课将详细介绍MCP协议的核心机制与设计思想,帮助我们理解如何通过这一…

作者头像 李华