惊帆JF141心率血氧模块简单使用(STM32标准库代码)

2023-10-15 05:30

本文主要是介绍惊帆JF141心率血氧模块简单使用(STM32标准库代码),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我的那个模组长这个样子

从它的外观上也可以看出,是用下面的这个模块采集心率和血氧数据,并通过上方的串口接口将采集到的数据发送出去的。

让我们看一下产品规格书

和我们的设想是一致的,所以通过此处,大概也就知道了,应该如何配置单片机的串口来实现通信了

这里我用的是PA9和PA10串口1

并且开启接收中断,用于及时的接收数据

void XU_Yang_Init(void)
{//开启GPIOA和USART1的时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//配置PA9GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 ;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10 ;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);//初始化USART1USART_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate = 38400;//配置波特率(一般选择9600/115200等常用值)USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//选择是否启用硬件流控制()此处选择的是不开启硬件流USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;//选择工作模式(此处选择的是发送和接收模式)USART_InitStructure.USART_Parity = USART_Parity_No;//选择奇偶校验模式(此处选择的是无校验)USART_InitStructure.USART_StopBits = USART_StopBits_1;//选择停止位数(此处选择的是一位停止位)USART_InitStructure.USART_WordLength = USART_WordLength_8b;//选择数据位数(此处选择的是8位,因为没有校验位(不包含停止位和起始位))USART_Init(USART1,&USART_InitStructure);USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//使能中断(选择的是接收数据寄存器不为空中断)//配置优先级分组NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//初始化NVICNVIC_InitTypeDef NVIC_Initstructure;NVIC_Initstructure.NVIC_IRQChannel = USART1_IRQn;//选择USART1通道NVIC_Initstructure.NVIC_IRQChannelCmd = ENABLE;//开启选择的IRQn通道NVIC_Initstructure.NVIC_IRQChannelPreemptionPriority = 0;//设置抢占优先级NVIC_Initstructure.NVIC_IRQChannelSubPriority = 0;//响应优先级NVIC_Init(&NVIC_Initstructure);//开启USARTUSART_Cmd(USART1,ENABLE);
}

再往下看就是这个模组怎么用串口使用了,当然就是类似于WIFI模块的AT指令那样,给模组发一个指令来让模组工作或者停止工作。让我们看看产品说明书上怎么写的

看到这里我们也就清楚了,模组的指令分为三种分别是采集、体检、休眠。

在这里我们想要的是简单使用所以只关注最基础的采集指令就可以了。

当然了,笔者在看规格书的时候也顺便看了一点,体检指令就是集中采集数据并上传到云端进行分析,但是云端这个解析并没有在官网上面找到,所以我也就没有过多关注了,休眠功能的话,我建议将采集指令学习明白之后再尝试着加入休眠指令,要不然不就本末倒置了嘛!

我们知道了开关指令,当然就可以通过STM32的串口来发送指令达到效果啦!

需要指出的是在实际使用中虽然模组能近乎百分之百的响应指令,但是发送指令之后建议延时600ms左右(可以直接写到XU_Yang_SendByte(uint8_t Byte)函数中,笔者之前忘记了,大家自己添加一下吧),等待模组进入工作状态再进行读取

void XU_Yang_SendByte(uint8_t Byte)
{USART_SendData(USART1,Byte);while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);/*发送数据寄存器空标志位;若为RESET则发送数据寄存器非空,此时不能继续向该寄存器继续写数据,如果发送数据寄存器非空继续写数据,就会覆盖之前的数据导致发送的数据丢失一部分;*/
}
uint8_t XU_Yang_GetRxFlag(void)
{if(XU_Yang_RxFlag==1){return 1;}return 0;
}

好了言归正传既然我们知道了采集的开关指令,那么我们怎么知道接收的哪些是我们想要的心率和血样数据呢?

最简单的办法当然就是继续看产品规格书啦!

从中我们就可以了解到采集开启之后,模组会通过串口发送以0xFF为包头的实时数据包,里面不仅包括了心率和血氧数据,也包括了心跳的原始数据和微循环等数据,我们的目的是读取心率和血氧,所以重点读取这两个数据就可以了。

所以我们既然知道了接受的数据长什么样子,需要的数据在哪个位置,那么我们就可以来编写接收中断函数了。

这段代码的思路是检测到包头开始接受,接收到最大数量重新开始检测包头,如此循环接收数据并将数据放在数组里面方便调用。

串口代码是参考江科大的,所以也用了状态机的思路。

从产品规格书上我们可以看到,总共76个字节,但我们需要的只有仅仅两个字节,所以我就想着定义一个只有两个字节的数组直接用于接收需要的数据,但是使用的时候发现,干扰数据有点多,所以又选取了前两个心跳数据来用于数据有效性检测。当然这可能不是必须的,大家也不必在这里纠结过多。

//在中断实现接收一个HEX数据包
void USART1_IRQHandler(void)
{static uint8_t RxState;//用来记录状态数据static uint8_t pRxPacket;//用来记录接收了几个数据if (USART_GetITStatus(USART1, USART_IT_RXNE) == SET)//再次确认标志位是否正确置位{uint8_t RxData = USART_ReceiveData(USART1);//读取接收数据寄存器中的数据,存储到uint8_t RxData中if(RxState == 0)//状态零(接收包头){if(RxData == 0xFF){RxState = 1;pRxPacket=2;}}else if(RxState == 1)//状态1(接收数据){	if(pRxPacket>=66 && pRxPacket<68){//将接收到的心率和血氧数据放在数组后两位XU_Yang_RxPacket[pRxPacket-64]=RxData;pRxPacket++;}else if(pRxPacket<2){//将接收到的监测数据有效性的数据放在数组前两位XU_Yang_RxPacket[pRxPacket]=RxData;pRxPacket++;}else if(pRxPacket>=2 && pRxPacket<66){pRxPacket++;}else if(pRxPacket>=68){pRxPacket=0;RxState=0;XU_Yang_RxFlag = 1;//将接收信号位置1,代表数据接收完成,可以读取}}USART_ClearITPendingBit(USART1, USART_IT_RXNE);//清除接收数据寄存器非空标志位}
}

到这里,我们从中断中就接收到了需要的数据,但是这些数据并不能直接使用,因为当你不放置手指时此时数据肯定是无效的,尽管你的手指放在上面,由于各种因素刚开始产生的数据当然也是我们不想要看到的,大家可以尝试一个下,用串口模块连接血氧模块,进行测试,会发现没有放手指时,心跳数据一般都是0XC4,当放置手指之后,按照规格书的说明读取对应位的数据也会发现这些数据有些违背常识。我们一般来说写一个驱动函数的目的就是方便主函数的调用,最好是那种,主函数一调用,就能知道准确的数据。所以以此想法为思路,我们便可以写一下我们的读取函数了。

首先就是读取数据的间隔,我们通过上一个图可以看出,实时数据包每隔1.64s发送一次,所以理论上我们对就收数据的调用间隔也应该是1.64s,但是我习惯让单片机找点事情做,所以就没有通过延时调整时间间隔,而是让单片机一直筛选数据,(当然了可能也有弊端,比如接收的数据只接收了前两位,然后刚好单片机新一轮的筛选数据开始进行,就会导致,前两位是这一次的数据,后两位是上一次的数据,就会导致这一次的数据被当成错误的给筛除掉,当然了我觉得我这样做的也是有好处的,既然正确的都能给筛除掉,那我读到数据的时候自然就是模组稳定之后的数据啦!哈哈哈哈)

说一下我筛出数据的思路:就是通过判断数组的前两位和后两位的约束条件判断是否采集成功。

前两位为心跳数据,当为0xC4时可以判定为采集失败,当后两位心率和血氧数据不在正常的范围同理也可以判断为采集失败,

在下面的代码中也可以看到我里面将发送开始与结束采集的命令注释掉了,因为本来想着读完数据之后就结束采集,但是,发现频繁的开关导致模组反应不及时,所以索性,就在主函数控制开启和关闭采集了。

/*** @brief       过滤无效数据,并返回数据是否采集成功(无效数据过多,代表采集失败,最多等待200次数据包),成功采集一次就跳出采集程序                * @param       无                  
* @retval       1 数据采集成功;0数据采集失败         */uint8_t XU_Yang_Receive(void)
{uint8_t  num=0;//接收到几个数据包
//	XU_Yang_SendByte(0x8A);//开始采集while(1){if(XU_Yang_GetRxFlag()==1){
////过滤无效数据if(XU_Yang_RxPacket[0]==0xC4 | XU_Yang_RxPacket[1]==0xC4 | (XU_Yang_RxPacket[2]<60 | XU_Yang_RxPacket[2]>130) | (XU_Yang_RxPacket[3]<80 | XU_Yang_RxPacket[3]>99) ){//数据无效,将无效数据清除for(uint8_t i=0;i<4;i++){XU_Yang_RxPacket[i]=0;}num++;//失败次数过多,采集失败if(num >200){delay_ms(10);//两次采集时间间隔过短导致了一直采集失败//原因未知//num=0;
//				XU_Yang_SendByte(0x88);//结束采集return 0;}}else//采集成功{num=0;
//			XU_Yang_SendByte(0x88);//结束采集return 1;}}}}

模块文件写好之后,主函数就简单多了

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "XU_Yang.h"
#include "Key.h"
uint8_t n;
//uint32_t u=0;
uint16_t s=650;
int main()
{/*现象:将手指放上去,刚开始时显示失败,等待大约15s左右,显示采集成功,采集成功后,数据更新也会加快,*/OLED_Init();XU_Yang_Init();OLED_ShowString(1,1,"RxPacket:");OLED_ShowString(2,1,"000ci   00%");XU_Yang_SendByte(0x8A);//开始采集while(1){while(s--){n = XU_Yang_Receive();}if(n==1){OLED_ShowString(3,1,"      ");OLED_ShowNum(2,1,XU_Yang_RxPacket[2],3);XU_Yang_RxPacket[2] = 0;OLED_ShowNum(2,9,XU_Yang_RxPacket[3],3);XU_Yang_RxPacket[3] = 0;OLED_ShowString(3,1,"Right!");}else if(n==0){OLED_ShowNum(2,1,0,3);OLED_ShowNum(2,9,0,3);OLED_ShowString(3,1,"      ");OLED_ShowString(3,1,"Lose!");}}
}​

下面时实际效果,用到的工程代码,和产品规格书我都会上传上去,有需要请自取哈

这篇关于惊帆JF141心率血氧模块简单使用(STM32标准库代码)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/215629

相关文章

如何使用celery进行异步处理和定时任务(django)

《如何使用celery进行异步处理和定时任务(django)》文章介绍了Celery的基本概念、安装方法、如何使用Celery进行异步任务处理以及如何设置定时任务,通过Celery,可以在Web应用中... 目录一、celery的作用二、安装celery三、使用celery 异步执行任务四、使用celery

使用Python绘制蛇年春节祝福艺术图

《使用Python绘制蛇年春节祝福艺术图》:本文主要介绍如何使用Python的Matplotlib库绘制一幅富有创意的“蛇年有福”艺术图,这幅图结合了数字,蛇形,花朵等装饰,需要的可以参考下... 目录1. 绘图的基本概念2. 准备工作3. 实现代码解析3.1 设置绘图画布3.2 绘制数字“2025”3.3

Jsoncpp的安装与使用方式

《Jsoncpp的安装与使用方式》JsonCpp是一个用于解析和生成JSON数据的C++库,它支持解析JSON文件或字符串到C++对象,以及将C++对象序列化回JSON格式,安装JsonCpp可以通过... 目录安装jsoncppJsoncpp的使用Value类构造函数检测保存的数据类型提取数据对json数

python使用watchdog实现文件资源监控

《python使用watchdog实现文件资源监控》watchdog支持跨平台文件资源监控,可以检测指定文件夹下文件及文件夹变动,下面我们来看看Python如何使用watchdog实现文件资源监控吧... python文件监控库watchdogs简介随着Python在各种应用领域中的广泛使用,其生态环境也

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

SpringCloud集成AlloyDB的示例代码

《SpringCloud集成AlloyDB的示例代码》AlloyDB是GoogleCloud提供的一种高度可扩展、强性能的关系型数据库服务,它兼容PostgreSQL,并提供了更快的查询性能... 目录1.AlloyDBjavascript是什么?AlloyDB 的工作原理2.搭建测试环境3.代码工程1.

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

springboot整合 xxl-job及使用步骤

《springboot整合xxl-job及使用步骤》XXL-JOB是一个分布式任务调度平台,用于解决分布式系统中的任务调度和管理问题,文章详细介绍了XXL-JOB的架构,包括调度中心、执行器和Web... 目录一、xxl-job是什么二、使用步骤1. 下载并运行管理端代码2. 访问管理页面,确认是否启动成功

Java中ArrayList的8种浅拷贝方式示例代码

《Java中ArrayList的8种浅拷贝方式示例代码》:本文主要介绍Java中ArrayList的8种浅拷贝方式的相关资料,讲解了Java中ArrayList的浅拷贝概念,并详细分享了八种实现浅... 目录引言什么是浅拷贝?ArrayList 浅拷贝的重要性方法一:使用构造函数方法二:使用 addAll(