news 2026/4/15 10:41:25

嵌入式面试问题: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/4/14 18:44:12

26、全功能应用:拼写检查与索引生成

全功能应用:拼写检查与索引生成 在文本处理领域,拼写检查和索引生成是两项重要的任务。下面将详细介绍如何使用相关工具和脚本完成这些任务。 拼写检查脚本 拼写检查脚本通过设置一个 shell 变量 AWKLIB 来指定 spellcheck.awk 脚本的位置。符号 “$*” 会展开为脚本名…

作者头像 李华
网站建设 2026/4/11 18:43:42

4.1 AI代码研究方法:快速掌握大型开源项目核心代码库

4.1 AI代码研究方法:快速掌握大型开源项目核心代码库 在现代软件开发中,理解和掌握大型开源项目的代码库是一项关键技能。传统的代码阅读方法往往效率低下且容易迷失在复杂的代码结构中。本节将介绍如何利用AI工具(如Cursor、Claude Code等)来快速、系统地解构和理解大型开…

作者头像 李华
网站建设 2026/4/6 4:19:22

4.1 AI赋能代码研究:快速解构大型开源项目

4.1 AI赋能代码研究:快速解构大型开源项目 在软件开发过程中,我们经常需要理解和学习大型开源项目的代码结构和实现原理。传统的代码阅读方式往往效率低下,难以快速把握项目的整体架构和核心逻辑。本节课将介绍如何利用AI工具(特别是Cursor和Claude Code)来快速解构和理解…

作者头像 李华
网站建设 2026/4/10 21:39:58

魔改YOLO13高阶版改进之结合C3k2与DySnakeConv电信天线设备检测

1. 魔改YOLO13高阶版改进之结合C3k2与DySnakeConv电信天线设备检测 1.1. 🚀 前言 在当今5G和物联网快速发展的时代,电信天线设备的检测与维护变得越来越重要!🔍 传统的检测方法效率低下且容易出错,而基于深度学习的目…

作者头像 李华
网站建设 2026/4/9 22:29:24

水下3D重建技术突破:SeaThru-NeRF折射校正与散射消除深度解析

水下3D重建技术突破:SeaThru-NeRF折射校正与散射消除深度解析 【免费下载链接】nerfstudio A collaboration friendly studio for NeRFs 项目地址: https://gitcode.com/GitHub_Trending/ne/nerfstudio 你是否曾困惑于水下拍摄的照片在3D重建时总是模糊不清&…

作者头像 李华