void setup()
{
pinMode(oxiInt, INPUT); // pin D0 connects to the interrupt output pin of the MAX30102
u8g2.begin();
u8g2.enableUTF8Print();
u8g2.clearBuffer();
delay(2000);
u8g2.setFont(u8g2_font_unifont_tr);
u8g2.setCursor(0, 14);
u8g2.print("Initializing...");
u8g2.sendBuffer();
Wire.begin();
Serial.begin(115200);
Serial.println("Initializing");
maxim_max30102_reset(); // resets the MAX30102
maxim_max30102_read_reg(REG_INTR_STATUS_1, &uch_dummy); // Reads/clears the interrupt status register
maxim_max30102_init(); // initialize the MAX30102
old_n_spo2 = 0.0;
Serial.println(F("Time[s]\tSpO2\tHR\tRatio\tCorr"));
timeStart = millis();
}
在 setup()
函数中,首先将 oxiInt
引脚设置为输入模式(用于连接到 MAX30102 的中断输出引脚),然后对 OLED 屏幕进行初始化操作,在开机画面上输出 “Initializing…”。接着使用 Wire
库初始化 I2C 总线,打开串口输出,并分别进行了 MAX30102 的复位、中断状态清除和初始化操作。最后输出表头 “Time[s]\tSpO2\tHR\tRatio\tCorr”,并记录开始时间。
显示心率和血氧值
void print_hr_spo2(int val_hr, int val_spo2){
if(val_hr > 999 || val_hr < 0){
return;
}
if(val_spo2 > 100 || val_spo2 < 0){
return;
}
u8g2.clearBuffer();
u8g2.setFont(u8g2_font_fub14_tr);
u8g2.setCursor(0,14);
u8g2.print(" HR");
u8g2.setCursor(60,14);
u8g2.print("SPO2");
char str_hr[4];
u8g2.setFont(u8g2_font_fur20_tr);
itoa(val_hr, str_hr, 10);
if(val_hr > 99){
u8g2.drawStr(0, 38, str_hr);
}else if(val_hr > 9){
u8g2.drawStr(14,38, str_hr);
}else{
u8g2.drawStr(28,38, str_hr);
}
char str_spo2[4];
u8g2.setFont(u8g2_font_fur30_tr);
itoa(val_spo2, str_spo2, 10);
if(val_spo2 > 99){
u8g2.drawStr(55, 48, str_spo2);
}else if(val_spo2 > 9){
u8g2.drawStr(70, 48, str_spo2);
}else{
u8g2.drawStr(85, 48, str_spo2);
}
u8g2.setFont(u8g2_font_fur11_tr);
u8g2.setCursor(13,50);
u8g2.print("bpm");
u8g2.setCursor(110,64);
u8g2.print("%");
u8g2.sendBuffer();
}
在 print_hr_spo2()
函数中,首先判断心率和血氧值是否在有效范围内(0~999 和 0~100),然后清空 OLED 屏幕缓存,并设置字体和显示位置。接着分别将心率和血氧值转化为字符串类型,并根据值的长度和位置绘制在屏幕上,最后输出单位并发送缓存。
void print_measuring(){
u8g2.clearBuffer();
u8g2.setFont(u8g2_font_fub14_tr);
u8g2.setCursor(0,14);
u8g2.print(" HR");
u8g2.setCursor(60,14);
u8g2.print("SPO2");
char str_hr[4];
u8g2.setFont(u8g2_font_fur20_tr);
u8g2.drawStr(10, 38, "---");
char str_spo2[4];
u8g2.setFont(u8g2_font_fur30_tr);
u8g2.drawStr(65, 48, "---");
u8g2.setFont(u8g2_font_unifont_tr);
u8g2.setCursor(26,62);
u8g2.print("Measuring...");
u8g2.sendBuffer();
}
在 print_measuring()
函数中,清空 OLED 屏幕缓存,并设置字体和显示位置。接着绘制未测量状态的心率和血氧值,并输出 “Measuring…”,最后发送缓存。
void print_press(){
u8g2.clearBuffer();
u8g2.setFont(u8g2_font_fub14_tr);
u8g2.setCursor(0,14);
u8g2.print(" HR");
u8g2.setCursor(60,14);
u8g2.print("SPO2");
char str_hr[4];
u8g2.setFont(u8g2_font_fur20_tr);
u8g2.drawStr(10, 38, "---");
char str_spo2[4];
u8g2.setFont(u8g2_font_fur30_tr);
u8g2.drawStr(65, 48, "---");
u8g2.setFont(u8g2_font_unifont_tr);
u8g2.setCursor(20,62);
u8g2.print("Keep pressing");
u8g2.sendBuffer();
}
在 print_press()
函数中,清空 OLED 屏幕缓存,并设置字体和显示位置。接着绘制要求将指尖放到传感器上。
完整代码如下:
#include <U8g2lib.h>
#include <SPI.h>
#include <Wire.h>
#include "algorithm_by_RF.h"
#include "max30102.h"
#define REVERSE_LED
U8G2_SSD1306_128X64_NONAME_F_SW_I2C u8g2(U8G2_R0, /* clock=*/ SCL, /* data=*/ SDA, /* reset=*/ U8X8_PIN_NONE);
const byte oxiInt = 16; // pin connected to MAX30102 INT
uint32_t elapsedTime, timeStart;
uint32_t aun_ir, aun_red;
uint32_t aun_ir_buffer[BUFFER_SIZE]; // infrared LED sensor data
uint32_t aun_red_buffer[BUFFER_SIZE]; // red LED sensor data
float old_n_spo2; // Previous SPO2 value
uint8_t uch_dummy;
bool showMeasuring = false;
void setup()
{
pinMode(oxiInt, INPUT); // pin D0 connects to the interrupt output pin of the MAX30102
u8g2.begin();
u8g2.enableUTF8Print();
u8g2.clearBuffer();
delay(2000);
u8g2.setFont(u8g2_font_unifont_tr);
u8g2.setCursor(0, 14);
u8g2.print("Initializing...");
u8g2.sendBuffer();
Wire.begin();
Serial.begin(115200);
Serial.println("Initializing");
maxim_max30102_reset(); // resets the MAX30102
maxim_max30102_read_reg(REG_INTR_STATUS_1, &uch_dummy); // Reads/clears the interrupt status register
maxim_max30102_init(); // initialize the MAX30102
old_n_spo2 = 0.0;
Serial.println(F("Time[s]\tSpO2\tHR\tRatio\tCorr"));
timeStart = millis();
}
void print_hr_spo2(int val_hr, int val_spo2){
if(val_hr > 999 || val_hr < 0){
return;
}
if(val_spo2 > 100 || val_spo2 < 0){
return;
}
u8g2.clearBuffer();
u8g2.setFont(u8g2_font_fub14_tr);
u8g2.setCursor(0,14);
u8g2.print(" HR");
u8g2.setCursor(60,14);
u8g2.print("SPO2");
char str_hr[4];
u8g2.setFont(u8g2_font_fur20_tr);
itoa(val_hr, str_hr, 10);
if(val_hr > 99){
u8g2.drawStr(0, 38, str_hr);
}else if(val_hr > 9){
u8g2.drawStr(14,38, str_hr);
}else{
u8g2.drawStr(28,38, str_hr);
}
char str_spo2[4];
u8g2.setFont(u8g2_font_fur30_tr);
itoa(val_spo2, str_spo2, 10);
if(val_spo2 > 99){
u8g2.drawStr(55, 48, str_spo2);
}else if(val_spo2 > 9){
u8g2.drawStr(70, 48, str_spo2);
}else{
u8g2.drawStr(85, 48, str_spo2);
}
u8g2.setFont(u8g2_font_fur11_tr);
u8g2.setCursor(13,50);
u8g2.print("bpm");
u8g2.setCursor(110,64);
u8g2.print("%");
u8g2.sendBuffer();
}
void print_measuring(){
u8g2.clearBuffer();
u8g2.setFont(u8g2_font_fub14_tr);
u8g2.setCursor(0,14);
u8g2.print(" HR");
u8g2.setCursor(60,14);
u8g2.print("SPO2");
char str_hr[4];
u8g2.setFont(u8g2_font_fur20_tr);
u8g2.drawStr(10, 38, "---");
char str_spo2[4];
u8g2.setFont(u8g2_font_fur30_tr);
u8g2.drawStr(65, 48, "---");
u8g2.setFont(u8g2_font_unifont_tr);
u8g2.setCursor(26,62);
u8g2.print("Measuring...");
u8g2.sendBuffer();
}
void print_press(){
u8g2.clearBuffer();
u8g2.setFont(u8g2_font_fub14_tr);
u8g2.setCursor(0,14);
u8g2.print(" HR");
u8g2.setCursor(60,14);
u8g2.print("SPO2");
char str_hr[4];
u8g2.setFont(u8g2_font_fur20_tr);
u8g2.drawStr(10, 38, "---");
char str_spo2[4];
u8g2.setFont(u8g2_font_fur30_tr);
u8g2.drawStr(65, 48, "---");
u8g2.setFont(u8g2_font_unifont_tr);
u8g2.setCursor(20,62);
u8g2.print("Keep pressing");
u8g2.sendBuffer();
}
void loop()
{
// Serial.println("looping");
float n_spo2, ratio, correl; // SPO2 value
int8_t ch_spo2_valid; // indicator to show if the SPO2 calculation is valid
int32_t n_heartrate; // heart rate value
int8_t ch_hr_valid; // indicator to show if the heart rate calculation is valid
int32_t i;
// buffer length of BUFFER_SIZE stores ST seconds of samples running at FS sps
// read BUFFER_SIZE samples, and determine the signal range
for (i = 0; i < BUFFER_SIZE; i++)
{
while (digitalRead(oxiInt) == 1)
; // wait until the interrupt pin asserts
delay(1);
// wdt_reset();
#ifdef REVERSE_LED
## 最后
**自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**
**深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。**
**因此收集整理了一份《2024年嵌入式&物联网开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。**







**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上嵌入式&物联网开发知识点,真正体系化!**
[**如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!**](https://bbs.csdn.net/topics/618654289)
**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**!!
链图片转存中...(img-rD7TX5Oi-1715627360035)]
[外链图片转存中...(img-ky5ouJft-1715627360036)]
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上嵌入式&物联网开发知识点,真正体系化!**
[**如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!**](https://bbs.csdn.net/topics/618654289)
**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**!!