找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 447|回复: 3
打印 上一主题 下一主题
收起左侧

单片机程序 心率数据显示不对

[复制链接]
跳转到指定楼层
楼主
ID:1142165 发表于 2025-3-31 16:44 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
#include <reg52.h>
#include <intrins.h>
#include <stdio.h>

#define uint unsigned int
#define uchar unsigned char
#define DASE 60000

// 定义 LM016L 引脚
sbit RS = P2^7;
sbit RW = P2^6;
sbit EN = P2^5;
// 定义 DS18B20 引脚
sbit DQ = P1^0;
// 定义心率引脚
sbit HeartRateSensor = P3^3;
#define LCD_DATA P0
// 定义蜂鸣器引脚
sbit BUZZER = P1^7;
// 定义MAX30102的SDA引脚
sbit SDA = P3^4;
sbit INT = P3^2;

// 温度上下限设置
#define TEMP_LOWER 35
#define TEMP_UPPER 40

// 函数声明
void delay_us(unsigned int us);
void delay_ms(unsigned int ms);
void lcd_init();
void lcd_command(unsigned char cmd);
void lcd_data(unsigned char dat);
void lcd_string(const char *str);
void display_heart_rate(unsigned int heart_rate);
void display_temperature(float temperature);
// DS18B20 相关函数声明
bit ds18b20_reset(void);
void ds18b20_write_byte(unsigned char dat);
unsigned char ds18b20_read_byte(void);
float ds18b20_read_temp(void);

uint  D_num;            // 定时器中断计数变量
// 声明 str 数组为全局变量
static unsigned char str[10];

// 全局变量用于心率计算
unsigned int pulse_count = 0;
unsigned int prev_pulse_count = 0;
unsigned int heart_rate = 0;
uint  temp;                   // 临时变量
uint  count; // 计数
float tempF;
bit   flag_S = 0;

// 定时器0初始化函数
void InitTimer0(void)
{
    TMOD = 0x01;
    TH0 = 0xFC;
    TL0 = 0x18;
    EA = 1;
    ET0 = 1;
    TR0 = 1;
}

void delay_ms(unsigned int ms) // @12MHz
{
    unsigned i, j;
    for (i = ms; i > 0; i--)
        for (j = 120; j > 0; j--);
}

// 初始化 LCD
void lcd_init()
{
    delay_ms(15);
    lcd_command(0x38); // 设置 16x2 显示,5x7 点阵,8 位数据
    delay_ms(5);
    lcd_command(0x38);
    delay_ms(5);
    lcd_command(0x38);
    lcd_command(0x0C); // 打开显示,关闭光标
    lcd_command(0x06); // 光标右移,字符不移动
    lcd_command(0x01); // 清屏
    delay_ms(5);
}

// 发送 LCD 命令
void lcd_command(unsigned char cmd)
{
    RS = 0;
    RW = 0;
    LCD_DATA = cmd;
    delay_ms(1);
    EN = 1;
    _nop_();
    EN = 0;
}

// 发送 LCD 数据
void lcd_data(unsigned char dat)
{
    RS = 1;
    RW = 0;
    LCD_DATA = dat;
    delay_ms(1);
    EN = 1;
    _nop_();
    EN = 0;
}

// 发送字符串到 LCD
void lcd_string(const char *str)
{
    while (*str)
    {
        lcd_data(*str++);
    }
}

// 显示心率
void display_heart_rate(unsigned int heart_rate)
{
    lcd_command(0x87); // 设置显示位置
    sprintf(str, "%d bpm ", heart_rate);
    lcd_string(str);
}

// 显示温度
void display_temperature(float temperature)
{
    lcd_command(0xC7); // 设置显示位置
    if (temperature == -1)
    {
        sprintf(str, "Err");
    }
    else
    {
        sprintf(str, "%.1f C", temperature);
    }
    lcd_string((const char *)str);
}

void delay(uint t) // @12T 1us
{
    while (t--);
}

void Init_DS18b20() // 18b20初始化函数
{
    DQ = 1;    // DQ复位
    delay(8);
    DQ = 0;    // 释放DQ
    delay(80); // DS18B20收到信号后延时
    DQ = 1;   // 读取DQ
    delay(30);
}

void write_byte(uchar dat) // 写时序
{
    uchar i = 0;
    for (i = 8; i > 0; i--)
    {
        DQ = 0;
        DQ = dat & 0x01; // 写1,在15微秒内拉低
        delay(5); // 写0,拉低60微秒
        DQ = 1;
        dat >>= 1;
    }
    delay(4);
}

uchar read_byte() // 读时序
{
    uchar i = 0;
    uchar dat = 0;
    for (i = 8; i > 0; i--)
    {
        DQ = 0; // 拉低ds18b20单总线
        dat >>= 1;
        DQ = 1; // 15微秒内拉释放总线
        if (DQ)
            dat |= 0x80;
        delay(4);
    }
    return (dat);
}

float ds18b20_read_temp(void)
{
    Init_DS18b20();  // 初始化18b20
    write_byte(0xcc); // 跳过读写序列号
    write_byte(0x44); // 启动温度转换
    Init_DS18b20();  // 初始化18b20
    write_byte(0xCC); // 跳过读写序列号
    write_byte(0xBE); // 读温度寄存器
    temp = read_byte(); // 读取温度低8位
    temp = read_byte() << 8 | temp; // 读取温度高8位
    tempF = temp * 0.0625;
    return tempF;
}

void Timer0Interrupt() interrupt 1
{
    static uint i;
    TH0 = 0xFC;
    TL0 = 0x18;
    count++;
    i++;
    if (i >= 500)                                // 0.5秒
    {
        i = 0;
        flag_S = 1;                        // 0.5秒标志
    }
}

// 外部中断 0 服务函数,用于检测心率脉冲
void External0_ISR() interrupt 0
{
    TR0 = 0;                                        // 关定时器0
    D_num = count;                        // 读取定时器1ms中断计数值
    count = 0;                                        // 定时器1ms中断计数值清0
    TH0 = 0xFC;
    TL0 = 0x18;
    TR0 = 1;                                        // 开定时器0
}

// 主程序
void main()
{
    float temperature = 0.0;
    unsigned int time_count = 0;
    // 初始化 LCD
    lcd_init();
    lcd_string("Heart: ");
    lcd_command(0xC0); // 第二行
    lcd_string("Temp: ");
    InitTimer0();
    // 初始化外部中断 0
    IT0 = 1;  // 下降沿触发
    EX0 = 1;  // 使能外部中断 0
    EA = 1;   // 使能全局中断
    IE0 = 0;         // INT0中断请求标志清0

    while (1)
    {
        if (flag_S) // 0.5秒
        {
            flag_S = 0;
            heart_rate = DASE / D_num;        // 计算心率
            display_heart_rate(heart_rate);
            // 获取温度
            temperature = ds18b20_read_temp();
            display_temperature(temperature);

            // 温度报警检查
            if (temperature < TEMP_LOWER || temperature > TEMP_UPPER)
            {
                BUZZER = 0;  // 蜂鸣器响
                delay_ms(500);  // 响一段时间
            }
            else
            {
                BUZZER = 1;  // 温度正常,关闭蜂鸣器
            }
        }
    }
}
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:67526 发表于 2025-4-17 09:12 | 只看该作者
sbit HeartRateSensor = P3^3;改为sbit HeartRateSensor = P3^2;或者改外中断号
回复

使用道具 举报

板凳
ID:67526 发表于 2025-4-17 09:14 | 只看该作者
sbit HeartRateSensor = P3^3;改为sbit HeartRateSensor = P3^2;或者在程序中改外相应的中断号
回复

使用道具 举报

地板
ID:1137639 发表于 2025-4-19 22:44 | 只看该作者
sbit HeartRateSensor = P3^3;改为sbit HeartRateSensor = P3^2;或者改外中断号
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表