news 2026/5/1 14:17:04

嵌入式面试问题: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/1 11:46:20

从16MHz/9pF讲起:晶振方向性、负载电容、脚位识别

无源晶振没有方向性。无源晶振通常会标注负载电容,例如6pF、9pF、12pF,因此当晶振标注为 “16MHz/9pF” 时,可以确定它是无源晶振,也就意味着它没有方向性。相反,有源晶振内部带有振荡芯片,会标注供电电压(…

作者头像 李华
网站建设 2026/5/1 9:04:19

CentOS-Stream-10 系统安装之Firewalld防火墙配置

安装完系统,配置完网络,selinux关掉后,还需要配置防火墙。这里以ssh服务为例,开启远程访问。firewall-cmd --add-servicessh --permanent firewall-cmd --reload--add-service 添加服务--permanent 永久生效--reload 重新加载或…

作者头像 李华
网站建设 2026/4/17 15:17:12

4.3 可视化表达利器:Mermaid流程图自动生成技巧

4.3 可视化表达利器:Mermaid流程图自动生成技巧 在前两节课中,我们学习了如何使用AI工具快速理解和掌握大型开源项目,并将代码理解转化为高质量的技术文档。本节课我们将专注于一个非常实用的技能——使用AI工具自动生成Mermaid流程图,帮助我们更直观地表达复杂的系统架构…

作者头像 李华
网站建设 2026/5/1 9:31:14

4.4 建立高效工作流:从代码研究到文档输出的完整流程

4.4 建立高效工作流:从代码研究到文档输出的完整流程 在前面的课程中,我们学习了如何使用AI工具进行代码研究、文档撰写和可视化生成。现在,我们将把这些技能整合起来,建立一个完整的高效工作流,从代码研究到文档输出,形成一个闭环的开发和知识管理流程。 完整工作流概…

作者头像 李华
网站建设 2026/5/1 9:30:04

TDengine压力测试

一、场景一个实例,1000个信号,100微秒采集一次数据1000x10x1000100万条/秒二、测试数据1、存储空间需求信号值为int100秒1亿条数据,插入批次大小为25000,数据大小为4MB纯数据压缩后每天:约4.42 GB每月:约13…

作者头像 李华