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

功能:
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/

  • 2
    点赞
  • 43
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 单片机数码管心率脉搏Proteus仿真是一种非常实用的工具,可以用于验证和测试数字电路设的精度和可靠性。通过这种仿真方式,工程师们可以快速而准确地评估数字电路设的性能,从而提高其设和开发的效率和效果。 在进行这种仿真时,首先需要准备一个数码管心率脉搏的电路设和对应的仿真模型。其中,电路设要充分考虑数字信号的传输、编码、解码和显示等各个环节,以保证信号的准确性和稳定性。 然后,将电路设导入Proteus仿真平台中,并进行各项参数设置和模拟仿真。通过对仿真结果的观察和分析,可以评估电路设的性能和可靠性,进一步优化设方案,提高系统的稳定性和精度。 总之,基于单片机数码管心率脉搏Proteus仿真是数字电路设和开发中非常常见的一种方法,对于提高系统性能和可靠性有着重要的作用,值得广大工程师掌握和应用。 ### 回答2: 单片机是一种重要的电子元器件,常用于控制系统中。数码管是一种显示器件,常用于数字显示。心率脉搏是测量人体健康状况的一种方式。Proteus是一种仿真软件,可用于电路图的设仿真。 基于单片机的数码管心率脉搏仿真是一种常见的电路设。该电路可以通过单片机记录用户的心率脉搏,并将其显示在数码管上。通过该设备,用户可以实时了解自己的心率脉搏情况,进而采取有针对性的健康行动。 在Proteus中进行仿真时,需要先设好电路图。电路图的设需要考虑到单片机的控制模块以及数码管的数字显示模块。在设电路图时,需要注意线路连接的正确性、电路的稳定性与可靠性等方面。 完成电路图设之后,还需进行仿真测试。在仿真过程中,可以通过Proteus提供的各种测试工具来测试电路的性能。例如,可以测试CPU的频率、数码管的显示效果等,以保证电路的正常工作。 综上所述,基于单片机的数码管心率脉搏仿真是一种有益的电路设。它可以对人体健康状况进行实时监测,提高用户的健康意识,对人们的健康管理起到积极的作用。 ### 回答3: 对于基于单片机数码管心率脉搏proteus仿真,我们需要首先确定使用的单片机。常见的单片机有AVR、STC、PIC等,其中STC单片机是国产的,价格实惠。在仿真之前,需要下载并安装Proteus仿真软件。 在Proteus中,我们可以通过搭建电路板来实现基于单片机数码管心率脉搏仿真。首先,我们需要根据单片机型号选购合适的元器件,比如显示器、滤波电容、电阻、晶振等。 在搭建电路板过程中,需要注意电路布局的合理性和连线质量的可靠性。功耗控制也是非常重要的一点,我们需要确保电路在运行中不会因为过高的功耗而损坏。 在电路板搭建完成后,我们就可以开始仿真了。在仿真过程中,需要注意观察单片机是否能够正确地控制显示器,显示出正确的心率脉搏数据。同时,还需要考虑单片机与其它元器件之间的通信和控制,确保电路能够稳定工作。 总的来说,基于单片机数码管心率脉搏仿真需要考虑许多因素,包括电路构建、元器件选型、功耗控制和仿真验证等,需要有一定的电子技术和Proteus仿真软件操作经验。但一旦成功,它能够极大地提高我们的电路设和调试效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值