RT-Thread 入门学习笔记:浮点数终端格式打印的四种实现方法

本文介绍了在RT-Thread操作系统中,如何实现浮点数的格式化终端打印。通过四种方法,包括修改系统代码、重写rt_kprintf、创建专用浮点打印函数以及使用ulog组件,详细讲解了浮点数打印的实现过程,并讨论了不同方法的优缺点。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

简介

  • RT-Thread 的rt_kprintf,普通的格式打印没有问题,但是暂不支持打印%lf、%f这种浮点打印
  • 如何格式化打印浮点数呢?如float类型,如double类型
  • 默认rt_kprintf格式化打印浮点数:
    rt_kprintf("RMC : Latitude: %lf[%c], Longitude: %lf[%c], Fix: %c\n", pack.lat, pack.ns,
        pack.lon, pack.ew,</
概述:这是一个数据采集的装置,本身没有什么亮点。主要是基于RT-Thread操作系统,驱动NB模块-BC26来实现数据的发送。值得一说的是RT-Thread本身有BC-26的驱动包。不过这里并没有使用,而是使用at-device软件包来驱动的BC26。因此稍微改一改内部的代码,就能驱动其他的AT设备。话回正题,我使用at-thread的目的就是驱动BC26建立TCP或UDP连接,使得板卡采集得到的数据能发送到我电脑上的TCP Server。当然,除了数据上传之外,也能实现上位机控制板卡。还有则是在代码中发现利用邮箱+消息队列来进行数据传输和通信真的很爽。 开发环境:硬件部分 ART-Pi (主控) BC-26 (NB-IOT模块) BHT11 (温湿度传感器) RT-Thread版本 RT-Thread V4.0.2 开发工具及版本 RT-Thread Studio V2.0.0 :RT-thread推出的IDE,免费。 Putty V0.73:开源免费的一款工具,我纯把他当成串口助手使用 花生壳 V5 :内网穿透工具。 网络调试助手(MetAssist V4.3.13):网上下的,应该比较出名。 RT-Thread使用情况描述:内核部分 调度器 消息队列 邮箱 组件部分 at_device UART 硬件框架描述先附图一张: 很简单的一个框架,总共只有主控,传感器,执行器,以及比较重要的云平台,这四大部分。传感器可以是任意传感器,只要发送的数值种类不一次性超出两种即可。执行器我在这里使用了板载的LED灯充当。云平台则是利用网络调试助手搭了一个TCP Server来充当。由于我个人没有固定外网IP,所以我如果直接使用网络助手,是无法将ART采集得到的数据传输到我的电脑上的。因此我利用花生壳将我的IP映射到了外网,使得板卡能连接到我创建的TCP Server上。 软件框架说明流程图如下: 本人并不是很会画流程图,所以辛苦大家看一看介绍吧。 其实在这个板卡中是要烧两套程序的,一套是bootloader负责初始化QSPI并且运行QSPI内的程序。所以这份程序是下载到片内Flash的。另一份则是具体的功能添加的比较多的程序。他是运行在QSPI中的。这两个程序必须先运行BootLoader否则QSPI中的程序是无法运行的。而由于BootLoader的职责是让程序从0x08000000跳转到0x90000000运行所以,如果QSPI中没有其他程序的话,Bootloader只会运行一次,表现的现象就是只打印一个LOGO。 其实在RT-Thread中其实有BC26的驱动包,可以直接拿来用,不需要自己再对BC26进行初始化,但是我这里使用的是at_device驱动包,所以自己要写一部分的代码,进行初始化。创建邮箱和消息队列则是为了两者相互配合一起实现发送同步消息的功能。 数据采集线程和数据发送线程之间使用消息队列+邮箱的方式实现消息同步,在这里数据采集线程可以有多个,而数据发送线程我这设立了一个。发送线程会将接收到的信息都发送到云平台中。 数据接收则是利用at_device中的代码实现的。利用内部的代码还可实现云平台发送消息控制板卡上的LED灯或者其他执行器。 软件模块说明消息队列+邮箱的消息同步方式 在使用消息队列+邮箱的方式来进行线程间消息同步的话需要先创建一个结构体,一个动态邮箱,一个消息队列。然后对结构体进行填充后利用消息队列发送出去,具体请看以下代码示例: //创建结构体部分 struct msg //消息队列发送此结构体的地址来实现线程间的同步 { char *str; int vol; float data1; int data2; struct rt_mailbox* ack; }; //创建动态邮箱部分 rt_mailbox_t mail_box1 = RT_NULL; //创建二氧化氮线程应答邮箱控制块 rt_mailbox_t mail_box2 = RT_NULL; //创建二氧化硫线程应答邮箱控制块 rt_mailbox_t mail_box3 = RT_NULL; //创建粉尘数据线程应答邮箱控制块 rt_mailbox_t mail_box4 = RT_NULL; //创建备用线程邮箱控制块 /**************创建多个应答邮箱******************/ int move_mail_box_sample(void) { mail_box1 = rt_mb_create("mail_box1", 1, RT_IPC_FLAG_FIFO); //创建动态邮箱1 mail_box2 = rt_mb_create("mail_box2", 4, RT_IPC_FLAG_FIFO); //创
### RT-Thread Studio 中使用 `rt_kprintf` 函数正确输出浮点数RT-Thread 操作系统中,默认情况下 `rt_kprintf` 并不支持 `%f` 这样的浮点数格式化选项,因为默认实现是经过裁剪的版本[^3]。 为了使 `rt_kprintf` 支持浮点数输出,可以采取以下几种方法之一: #### 方法一:启用完整的 printf 实现 可以通过配置 RT-Thread 的组件来启用完整版的 `printf` 功能。这通常涉及到修改 `.config` 文件中的相应选项或将特定宏定义加入编译器预处理器指令里。具体操作如下所示: ```c #define RT_USING_NEWLIB /* 启用 Newlib C 库 */ ``` 当启用了新库之后,在重新构建项目之前记得保存更改并同步到 IDE 当前工作区环境内。 #### 方法二:自定义 vsnprintf 函数 另一种解决方案是在应用程序层面上重写或扩展标准库里的字符串处理接口函数如 `vsnprintf()` 。这里提供了一个简单的例子说明如何通过替换原有弱声明符号的方式来自定义该功能[^2]: ```c #include <stdio.h> #include <stdarg.h> // 自定义实现了对 %f 格式的解析和支持 int my_rt_vsnprintf(char *str, size_t count, const char *format, va_list ap) { // ... (此处省略具体的实现细节) return result; } __attribute__((weak)) int rt_vsnprintf(char *buf, rt_size_t size, const char *fmt, va_list args) { return my_rt_vsnprintf(buf, size, fmt, args); } ``` 上述代码片段展示了怎样利用 GCC 提供的功能属性标签 (`__attribute__((weak))`) 来覆盖原生定义,并引入自己定制化的逻辑以满足需求。 完成以上任一步骤后,再次尝试调用 `rt_kprintf("%f", value);` 即可正常显示带有小数部分的数据了。 #### 示例程序展示 下面给出一段完整的测试代码用于验证是否成功解决了问题: ```c #include <rtthread.h> void test_float_output(void) { float pi = 3.1415926; rt_kprintf("Pi is approximately %.2f\n", pi); } MSH_CMD_EXPORT(test_float_output, Test floating point output with rt_kprintf); ``` 运行这段命令行工具插件后应该能够在终端界面上看到形似 “Pi is approximately 3.14”的预期结果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zhangsz_sh

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值