基于51单片机心率计脉搏体温测量仪WIFI无线通信proteus仿真原理图PCB

本系统采用STC89C52单片机实时监测佩戴者的心率和体温,并通过LCD1602显示数据。同时,系统通过WIFI定时发送数据,支持多连接并建立服务端口。

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

功能:
0.本系统采用STC89C52作为单片机
1.系统实时监测当前佩戴者的心率和体温,通过LCD1602显示相关数据
2.定时通过WIFI串口发送佩戴者的心率和体温
3.系统预留出串口接口,需要注意的是该型号的单片机只有一路串口,下载和WiFi通信是共用的该接口,为了防止冲突,再下载程序的时候需要把WiFi模块拔下来,下载完了之后取下下载器再插上WiFi模块!
4.采用DC002作为电源接口可直接输入5V给整个系统供电

原理图:
在这里插入图片描述

PCB :
在这里插入图片描述

主程序:

#include "reg52.h"
#include <stdio.h>
#include "delay.h"
#include "lcd1602.h"
#include "18b20.h"

unsigned char datIndex = 0;
unsigned int timeCnt = 0; //中间变量值
unsigned char heartRate = 0; 
unsigned int heartRateBuf[3] = {0};                                              //记录多次心率值求平均
unsigned char xdata dis0[16];                                            //定义显示区域临时存储数组
bit dispFlag = 0;                                                    //更新标志变量
bit readTempFlag = 0;                                                  //读取温度标志
unsigned char g_messageCnt;                                              //短信发送计数
bit refreshHRFlag = 0;
bit updateFlag = 0;
bit sendFlag = 0;


void InitTimer0(void); //函数声明
void UARTSendStr(unsigned char *s, unsigned char length);
void UARTInit(void);
void UARTSendByte(unsigned char dat);
void InitTimer1(void);

/****************主函数***************/
void main()
{
    int tempBuf;      //温度读取值
    float temperature; //温度实际值

    EX1 = 1;       //允许外部中断1中断
    PX1 = 1;
    EA = 1;        //开总中断
    IT1 = 1;       //外部中断1负跳变中断
    InitTimer0(); //初始化定时器
    InitTimer1();
    UARTInit();
    
    LCD_Init();  //初始化液晶
    DelayMs(200); //延时有助于稳定
    LCD_Clear(); //清屏
    DelayMs(200);

    LCD_DispStr(0, 0, " Please wait!!! ");
    LCD_DispStr(0, 1, "WIFI is starting");

    DelayS(2);    
        
    UARTSendStr("AT+CIPMUX=1\r\n", 13); //打开多连接
    DelayMs(250);
    UARTSendStr("AT+CIPSERVER=1,8080\r\n", 21); //建立服务 端口号为8080
    DelayMs(250);

    LCD_DispStr(0, 0, "   Welcome!!!   ");
    LCD_DispStr(0, 1, "WIFI is ready!!!");

    DelayS(2);    
    LCD_Clear();

    while (1)
    {
        if (refreshHRFlag == 1) //刷新心率
        {
            EX1 = 0;
            TR0 = 0;
            refreshHRFlag = 0;
            heartRateBuf[datIndex] = timeCnt * 5; //算出间隔时间
            TH0 = 0xEE; //(65536 - 0.005*11059200/12) / 256
            TL0 = 0x00; //(65536 - 0.005*11059200/12) % 256  5ms
            timeCnt = 0; //50ms计数清零
            datIndex++;
            if (datIndex > 3) //记录到超过等于2次时间
            {
                datIndex = 1;   //计数从1开始
                updateFlag = 1; //测得2次开始显示
            }
            EX1 = 1;
            TR0 = 1;
        }

        if (dispFlag == 1)
        {
            if (updateFlag == 0) // 如果显示关  检测心率是否真实
            {
                heartRate = 0;
            }
            else // 如果显示开
            {
                heartRate = 60000 / ((heartRateBuf[1] >> 1) + (heartRateBuf[2] >> 1)); // 心率计算 2次求平均值
            }
            // EA = 0;
            DS18B20_Start();           // 启动温度检测
            DS18B20_GetTemp(&tempBuf); // 获取温度中间值
            // EA = 1;
            temperature = (float)tempBuf * 0.0625; // 温度值转换
            sprintf(dis0, "Heart:%3dbpm", (unsigned int)heartRate); // 打印当前心率
            LCD_DispStr(0, 0, dis0); // 显示心率
            sprintf(dis0, "Temp:%5.1f", temperature); // 打印当前温度
            LCD_DispStr(0, 1, dis0);                  // 显示温度
            LCD_DispOneChar(10, 1, 0xdf);             // 显示℃
            LCD_DispOneChar(11, 1, 'C');              // 显示℃
            dispFlag = 0;                             // 更新标志清零
        }

        if (sendFlag == 1)                   //数据发送计数
        {
            sendFlag = 0;
            UARTSendStr("AT+CIPSEND=0,28\r\n", 17); //发送32位数据
            DelayMs(20);
            sprintf(dis0, "Heart:%3dbpm\r\n", (unsigned int)heartRate); // 发送当前心率
            UARTSendStr(dis0, 14); //发送数据内容
            DelayMs(20);
            sprintf(dis0, "Temp:%5.1f'C\r\n", temperature); // 发送温度
            UARTSendStr(dis0, 14); //发送数据内容
            DelayMs(20);
        }
    }
}

void EXT1() interrupt 2
{
    if (timeCnt < 60) //当连续两次检测时间间隔小于60*5ms=300ms不处理
    {
        TR0 = 1; //开定时器
    }
    else
    {
        refreshHRFlag = 1;
    }
}

/*************定时器0初始化程序***************/
void InitTimer0(void)
{
    TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
    TH0 = 0xEE; //(65536 - 0.005*11059200/12) / 256
    TL0 = 0x00; //(65536 - 0.005*11059200/12) % 256  5ms
    PT0 = 1;      //设置高优先级
    EA = 1;       //总中断打开
    ET0 = 1;      //定时器中断打开
    TR0 = 0;      //定时器开关关闭
}

/*************定时器0中断服务程序***************/
void Timer0_INT() interrupt 1
{
    TH0 = 0xEE; //(65536 - 0.005*11059200/12) / 256
    TL0 = 0x00; //(65536 - 0.005*11059200/12) % 256  5ms
    timeCnt++;        //每50ms一次计数
    if (timeCnt > 300) //当超过300*5ms=1.5s没有检测到信号停止显示
    {
        datIndex = 0;       //数据个数清零
        timeCnt = 0; //10ms计数清零
        updateFlag = 0; //显示关
        TR0 = 0;       //定时器关
    }
}

void InitTimer1(void)
{
    TMOD |= 0x10; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
	TL1 = 0x00;		//设置定时初始值
	TH1 = 0x4C;		//设置定时初始值 50ms
    EA = 1;       //总中断打开
    ET1 = 1;      //定时器中断打开
    TR1 = 1;      //定时器开关打开
}
void Timer1_INT(void) interrupt 3
{
    static unsigned char time_50ms = 0;

	TL1 = 0x00;		//设置定时初始值
	TH1 = 0x4C;		//设置定时初始值 50ms
    
    time_50ms++;
    if (time_50ms > 100)
    {
        sendFlag = 1; //5s发送一次数据
        time_50ms = 0;
    } 

    if (time_50ms % 20 == 0) //定时1s到
    {
        dispFlag = 1; //更新标志置位
    }
}

void UARTInit(void)
{
	SCON = 0x50; //串口方式1,1个起始位,1个停止位,8位数据,可变波特率

    //波特率115200
	TH2 = 0xFF;
	TL2 = 0xFD;
	RCAP2H = 0xFF;  //(65536-(FOSC/32/BAUD))   BAUD = 115200 FOSC = 11059200
	RCAP2L = 0xFD;
	
	/********波特率发生器接收和发送相同波特率,内部控制*********/
	TCLK = 1; //发送时钟标志
	RCLK = 1; //接收时钟标志
	C_T2 = 0; //内部定时器
	EXEN2 = 0; //关闭定时器2中断

	TR2 = 1; //打开定时器2计数
	ES  = 0; //关闭串口中断
	EA  = 1; //打开总中断
    // PS  = 1; //高优先级
}

void UARTSendByte(unsigned char dat)
{
    unsigned char time_out;
    time_out = 0x00;
    SBUF = dat;                       //将数据放入SBUF中
    while ((!TI) && (time_out < 100)) //检测是否发送出去
    {
        time_out++;
        DelayUs10x(2);
    }       //未发送出去 进行短暂延时
    TI = 0; //清除ti标志
}

void UARTSendStr(unsigned char *s, unsigned char length)
{
    unsigned char num = 0;
    while (num < length) //发送长度对比
    {
        UARTSendByte(*s); //放松单字节数据
        s++;              //指针++
        num++;            //下一个++
    }
}

// void UART_INT(void) interrupt 4 //串行中断服务程序
// {
//     if (RI) //判断是接收中断产生
//         RI = 0; //标志位清零
//     if (TI) //如果是发送标志位,清零
//         TI = 0;
// }

仿真演示视频:
https://www.bilibili.com/video/BV1H3411f7DN/

实物演示视频:
https://www.bilibili.com/video/BV1yR4y1f7Y5/

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值