news 2026/5/20 1:31:38

STM32F03C8T6通过AT指令获取天气API-下篇

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
STM32F03C8T6通过AT指令获取天气API-下篇

这里写目录标题

  • 堆栈大小换算与修改说明
    • 一、十六进制到十进制的换算
      • 1.1 换算公式
      • 1.2 换算结果
    • 二、修改前后的对比
      • 修改前(原始配置):
      • 修改后(新配置):
    • 三、STM32F103C8T6的内存限制
    • 四、为什么需要调整堆栈大小?
      • 4.1 原始设置(1KB栈+0.5KB堆)的问题:
      • 4.2 建议设置(4KB栈+8KB堆)的好处:
    • 五、完整的startup文件修改示例
    • 六、不同内存需求对应的推荐配置
    • 七、验证堆栈是否足够的方法
      • 7.1 在代码中添加堆栈使用检查
      • 7.2 在Keil/IDE中查看内存使用
    • 八、针对您之前报错的解决方案
    • 九、修改建议总结

堆栈大小换算与修改说明

一、十六进制到十进制的换算

1.1 换算公式

十六进制到十进制换算: 0x400 = 4 × 16² = 4 × 256 = 1024 0x1000 = 1 × 16³ = 1 × 4096 = 4096 0x200 = 2 × 16² = 2 × 256 = 512 0x2000 = 2 × 16³ = 2 × 4096 = 8192

1.2 换算结果

十六进制十进制字节KB
0x40010241024字节1KB
0x100040964096字节4KB
0x200512512字节0.5KB
0x200081928192字节8KB

二、修改前后的对比

修改前(原始配置):

  • Stack_Size = 0x400 = 1KB(栈大小)
  • Heap_Size = 0x200 = 0.5KB(堆大小)
  • 总计:1.5KB RAM

修改后(新配置):

  • Stack_Size = 0x1000 = 4KB(栈大小)
  • Heap_Size = 0x2000 = 8KB(堆大小)
  • 总计:12KB RAM

三、STM32F103C8T6的内存限制

STM32F103C8T6的RAM大小为20KB(0x5000字节)

内存区域地址范围大小
总RAM0x20000000-0x20004FFF20KB
栈(Stack)向下生长4KB(修改后)
堆(Heap)向上生长8KB(修改后)
全局/静态变量中间区域8KB(剩余)
内存布局示意图(修改后): 0x20005000 ↑ │ 栈(Stack) - 4KB (向下生长) │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ 0x20004000 ├─────────┤ │ 全局变量 │ │ 静态变量 │ │ 已分配内存 │ ├─────────┤ 0x20002000 ↑ │ 堆(Heap) - 8KB (向上生长) │ 0x20000000

四、为什么需要调整堆栈大小?

4.1 原始设置(1KB栈+0.5KB堆)的问题:

  1. 栈太小

    • 函数调用层次有限
    • 局部变量不能太大
    • 中断嵌套可能溢出
    • cJSON递归解析可能溢出
  2. 堆太小

    • cJSON动态分配内存不够
    • 网络数据缓冲区不足
    • 字符串操作受限
    • 无法使用标准库的malloc

4.2 建议设置(4KB栈+8KB堆)的好处:

  1. 栈(4KB)

    • 支持较深的函数调用
    • 可定义较大的局部数组
    • 中断处理更安全
    • 支持cJSON的递归解析
  2. 堆(8KB)

    • cJSON有足够内存解析完整JSON
    • 可创建较大的动态缓冲区
    • 支持标准库函数
    • 网络数据处理更灵活

五、完整的startup文件修改示例

; startup_stm32f103xe.s ; 堆栈配置部分 IF :DEF:__MICROLIB EXPORT __initial_sp EXPORT __heap_base EXPORT __heap_limit ELSE IMPORT __use_two_region_memory EXPORT __user_initial_stackheap __user_initial_stackheap LDR R0, = Heap_Mem LDR R1, =(Stack_Mem + Stack_Size) LDR R2, =(Heap_Mem + Heap_Size) LDR R3, = Stack_Mem BX LR ALIGN ENDIF ; ========================================================================== ; 修改前:Stack_Size EQU 0x00000400 ; 修改后:Stack_Size EQU 0x00001000 Stack_Size EQU 0x00001000 ; ← 修改这里,改为4KB AREA STACK, NOINIT, READWRITE, ALIGN=3 Stack_Mem SPACE Stack_Size __initial_sp ; ========================================================================== ; 修改前:Heap_Size EQU 0x00000200 ; 修改后:Heap_Size EQU 0x00002000 Heap_Size EQU 0x00002000 ; ← 修改这里,改为8KB AREA HEAP, NOINIT, READWRITE, ALIGN=3 __heap_base Heap_Mem SPACE Heap_Size __heap_limit PRESERVE8 THUMB

六、不同内存需求对应的推荐配置

应用场景栈大小堆大小总RAM需求说明
裸机简单应用1KB0.5KB1.5KB无动态分配,简单控制
中等复杂度2KB4KB6KB有动态分配,简单网络
JSON解析+网络4KB8KB12KB推荐配置
FreeRTOS简单4KB8KB12KB+多任务需要更多栈
FreeRTOS复杂6KB12KB18KB+接近芯片极限

七、验证堆栈是否足够的方法

7.1 在代码中添加堆栈使用检查

// 在main.c中添加堆栈使用检查externuint32_t__heap_base;externuint32_t__heap_limit;externuint32_t__initial_sp;externuint32_t__StackTop;externuint32_t__StackLimit;voidcheck_stack_heap_usage(void){// 获取当前栈指针registeruint32_t*stack_ptrasm("sp");// 计算栈使用量uint32_tstack_top=(uint32_t)&__StackTop;// 栈顶uint32_tstack_usage=stack_top-(uint32_t)stack_ptr;uint32_tstack_free=(uint32_t)stack_ptr-(uint32_t)&__StackLimit;// 计算堆使用量uint32_theap_base=(uint32_t)&__heap_base;uint32_theap_limit=(uint32_t)&__heap_limit;printf("=== 内存使用情况 ===\r\n");printf("栈大小: %lu bytes (%.1fKB)\r\n",Stack_Size,Stack_Size/1024.0);printf("栈已用: %lu bytes\r\n",stack_usage);printf("栈剩余: %lu bytes\r\n",stack_free);printf("堆大小: %lu bytes (%.1fKB)\r\n",Heap_Size,Heap_Size/1024.0);printf("堆起始: 0x%08lX\r\n",heap_base);printf("堆结束: 0x%08lX\r\n",heap_limit);printf("===================\r\n");}// 在main函数中定期调用intmain(void){// 初始化...while(1){check_stack_heap_usage();HAL_Delay(10000);// 每10秒检查一次}}

7.2 在Keil/IDE中查看内存使用

1. 编译后查看.map文件 2. 查找以下信息: Total RO Size (Code + RO Data) xxxx Total RW Size (RW Data + ZI Data) xxxx Total ROM Size (Code + RO Data + RW Data) xxxx 3. RW + ZI ≈ RAM使用量

八、针对您之前报错的解决方案

您之前的编译错误确实与startup文件有关,但不是堆栈设置的问题,而是:

  1. 中文字符编码问题:字符串中的中文字符导致编译器无法识别
  2. 解决方案:使用纯ASCII字符的JSON字符串进行测试
// 错误的(有中文字符):char*json_data="{\"name\":\"合肥\",\"weather\":\"阴\"}";// 正确的(纯ASCII):char*json_data="{\"name\":\"Beijing\",\"weather\":\"Cloudy\"}";

九、修改建议总结

  1. 必须修改

    • Stack_Size: 0x400 → 0x1000 (1KB → 4KB)
    • Heap_Size: 0x200 → 0x2000 (0.5KB → 8KB)
  2. 原因

    • cJSON需要动态内存分配
    • 网络数据需要缓冲区
    • 函数调用需要栈空间
  3. 注意事项

    • STM32F103C8T6只有20KB RAM
    • 修改后总共使用12KB,还有8KB给全局变量
    • 实际使用中监控堆栈使用情况
  4. 如果还不够用

    • 优化代码,减少栈使用
    • 使用静态分配代替动态分配
    • 减少递归深度
    • 优化网络缓冲区大小

这样修改后,您的JSON解析和网络通信应该就能正常工作了。

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

三维动态避障路径规划:基于部落竞争与成员合作算法(CTCM)融合动态窗口法DWA的无人机三维动态避障方法研究附MATLAB代码

✅作者简介:热爱科研的Matlab仿真开发者,擅长数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。🍎 往期回顾关注个人主页:Matlab科研工作室👇 关注我领取海量matlab电子书和数学建模资料 &#x1f34…

作者头像 李华
网站建设 2026/5/16 4:17:10

AI原生应用持续学习框架对比:TensorFlow vs PyTorch

AI原生应用持续学习框架对比:TensorFlow vs PyTorch 一、引言 (Introduction) 钩子 (The Hook) 你是否遇到过这样的困境? 花费数周训练的图像分类模型,上线3个月后准确率从92%暴跌至75%——原因是用户上传的图片中出现了新的类别(比如原本识别“猫/狗”,现在多了“仓鼠…

作者头像 李华
网站建设 2026/5/19 15:21:40

强烈安利9个AI论文写作软件,本科生论文无忧!

强烈安利9个AI论文写作软件,本科生论文无忧! 1.「千笔」—— 一站式学术支持“专家”,从初稿到降重一步到位(推荐指数:★★★★★)在论文写作过程中,很多本科生常常面临查重率高、内容重复、语言…

作者头像 李华
网站建设 2026/5/16 17:42:19

Gemini认证失败?一文搞定所有疑难

Gemini认证疑难解答会技术文章大纲认证前的准备工作确保已安装最新版本的Gemini客户端或SDK 检查系统环境是否符合Gemini运行要求 确认网络连接稳定且能访问Gemini服务端常见认证失败原因分析API密钥无效或过期导致认证被拒绝 请求频率超出配额限制触发系统保护 时间戳偏差过大…

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

Linux软件安装 —— JDK安装

文章目录一、节点说明二、下载安装包三、检查/删除现有JDK四、安装一、节点说明 IP主机名192.168.10.102node02192.168.10.103node03192.168.10.104node04二、下载安装包 官网地址:Java Archive Downloads - Java SE 8u211 and later | Oracle 中国 本文安装版本…

作者头像 李华
网站建设 2026/5/19 20:50:46

8. vLLM vs TensorRT-LLM

作者:HOS(安全风信子) 日期:2026-01-17 来源平台:GitHub 摘要: 2026年,vLLM和TensorRT-LLM是NVIDIA生态中最主流的两大推理框架。本文深入对比了vLLM与TensorRT-LLM的优劣,包括vLLM的灵活调度优势和TensorR…

作者头像 李华