基于51单片机的脉搏测量仪—心率计

基于51单片机的脉搏测量仪

(仿真+程序+原理图+设计报告)

功能介绍

具体功能:

本系统由STC89C51/52单片机+LCD1602显示模块+5mm红外接收管+LM358运放电路+按键模块等构成

1.手指放到红外对管中,2秒内读出心率;

2.LCD1602液晶显示当前的心率(心率/分钟);

3.按键可以设置报警的上下限心率,当测得的心率值超过报警值时发出报警。

​演示视频:

基于51单片机的脉搏测量仪—心率计

添加图片注释,不超过 140 字(可选)

程序

#include <reg52.h>
#include <intrins.h>	 //	包含头文件
/***公众号:木子单片机****/
#define uint            unsigned int
#define uchar           unsigned char
#define ulong           unsigned long	 //宏定义
#define LCD_DATA        P0				 //定义P0口为LCD_DATA

sbit LCD_RS =P2^5;
sbit LCD_RW =P2^6;
sbit LCD_E  =P2^7;						 //定义LCD控制引脚

sbit Xintiao =P1^0 ;					 //脉搏检测输入端定义
sbit speaker =P2^4;						 //蜂鸣器引脚定义

void delay5ms(void);   //误差 0us
void LCD_WriteData(uchar LCD_1602_DATA);	 /********LCD1602数据写入***********/
void LCD_WriteCom(uchar LCD_1602_COM);		 /********LCD1602命令写入***********/
void lcd_1602_word(uchar Adress_Com,uchar Num_Adat,uchar *Adress_Data); /*1602字符显示函数,变量依次为字符显示首地址,显示字符长度,所显示的字符*/
void InitLcd();//液晶初始化函数

void Tim_Init();

uchar Xintiao_Change=0;	   //
uint  Xintiao_Jishu;
uchar stop;
uchar View_Data[3];
uchar View_L[3];
uchar View_H[3];
uchar Xintiao_H=100;	//脉搏上限
uchar Xintiao_L=40;		//脉搏下限


uchar Key_Change;
uchar Key_Value;		//按键键值
uchar View_Con;			//设置的位(0正常工作,1设置上限,2设置下限)
uchar View_Change;

void main()	  //主函数
{
 InitLcd();
 Tim_Init();
 lcd_1602_word(0x80,16,"Heart Rate:     ");	  //初始化显示
 TR0=1;
 TR1=1;				  //打开定时器
 while(1)			  //进入循环
  {
   if(Key_Change)	  //有按键按下并已经得出键值
    {
	 Key_Change=0;	  //将按键使能变量清零,等待下次按键按下
	 View_Change=1;
	 switch(Key_Value)				//判断键值
	  {
	   case 1:						//设置键按下
	         {
			  View_Con++;			//设置的位加
			  if(View_Con==3)		//都设置好后将此变量清零
			   View_Con=0;
			  break;				//跳出,下同
			 }
	   case 2:						//加键按下
	         {
			  if(View_Con==2)		//判断是设置上限
			   {
			   	if(Xintiao_H<150)	//上限数值小于150
				 Xintiao_H++;		//上限+
			   }
			  if(View_Con==1)		//如果是设置下限
			   {
			   	if(Xintiao_L<Xintiao_H-1)//下限值小于上限-1(下限值不能超过上限)
				 Xintiao_L++;		//下限值加
			   }
			  break;
			 }
	   case 3:						//减键按下
	         {
			  if(View_Con==2)		//设置上限
			   {
			   	if(Xintiao_H>Xintiao_L+1)//上限数据大于下限+1(同样上限值不能小于下限)
				 Xintiao_H--;		//上限数据减
			   }
			  if(View_Con==1)		//设置下限
			   {
			   	if(Xintiao_L>30)	//下限数据大于30时
				 Xintiao_L--;		//下限数据减
			   }
			  break;
			 } 
	 }
	}
   if(View_Change)//开始显示变量
    {
	 View_Change=0;//变量清零
	 if(stop==0)			  //心率正常时
	  {
	   if(View_Data[0]==0x30) //最高位为0时不显示
	    View_Data[0]=' ';
	  }
	 else					  //心率不正常(计数超过5000,也就是两次信号时间超过5s)不显示数据
	  {
	   View_Data[0]=' ';
	   View_Data[1]=' ';
	   View_Data[2]=' ';
	  }

	 switch(View_Con)
	  {
	   case 0: //正常显示
	          {
			   lcd_1602_word(0x80,16,"Heart Rate:     ");//显示一行数据
			   lcd_1602_word(0xc0,16,"                ");//显示第二行数据
			   lcd_1602_word(0xcd,3,View_Data);			 //第二行显示心率
			   break;
			  }
	   case 1: //设置下限时显示
	          {
			   lcd_1602_word(0x80,16,"Heart Rate:     ");//第一行显示心率
			   lcd_1602_word(0x8d,3,View_Data);
			   
			   View_L[0]=Xintiao_L/100+0x30;		//将下限数据拆字
			   View_L[1]=Xintiao_L%100/10+0x30;
			   View_L[2]=Xintiao_L%10+0x30;

			   if(View_L[0]==0x30)					//最高位为0时,不显示
			    View_L[0]=' ';
			   
			   lcd_1602_word(0xC0,16,"Warning L :     ");//第二行显示下限数据
			   lcd_1602_word(0xCd,3,View_L);
			   break;
			  }
	   case 2: //设置上限时显示(同上)
	          {
			   lcd_1602_word(0x80,16,"Heart Rate:     ");
			   lcd_1602_word(0x8d,3,View_Data);
			   
			   View_H[0]=Xintiao_H/100+0x30;
			   View_H[1]=Xintiao_H%100/10+0x30;
			   View_H[2]=Xintiao_H%10+0x30;

			   if(View_H[0]==0x30)
			    View_H[0]=' ';
			   
			   lcd_1602_word(0xC0,16,"Warning H :     ");
			   lcd_1602_word(0xCd,3,View_H);
			   break;
			  }
	  }
	}
  }
}

void Time1() interrupt 3		//定时器1服务函数
{
 static uchar Key_Con,Xintiao_Con;
 TH1=0xd8;		   //10ms
 TL1=0xf0;		   //重新赋初值
 switch(Key_Con)   //无按键按下时此值为0
  {
   case 0:		   //每10ms扫描此处
          {
		   if((P3&0x07)!=0x07)//扫描按键是否有按下
		    {
			 Key_Con++;		  //有按下此值加1,值为1
			}
		   break;
		  }
   case 1:					  //10ms后二次进入中断后扫描此处(Key_Con为1)
          {
		   if((P3&0x07)!=0x07)//第二次进入中断时,按键仍然是按下(起到按键延时去抖的作用)
		    {
			 Key_Con++;		  //变量加1,值为2
			 switch(P3&0x07)  //判断是哪个按键按下
			  {
			   case 0x06:Key_Value=1;break;	 //判断好按键后将键值赋值给变量Key_Value
			   case 0x05:Key_Value=2;break;
			   case 0x03:Key_Value=3;break;
			  }
			}
		   else								 //如果10ms时没有检测到按键按下(按下时间过短)
		   	{
			 Key_Con=0;						 //变量清零,重新检测按键
			}
		   break;
		  }
   case 2:									 //20ms后检测按键
          {
		   if((P3&0x07)==0x07)				 //检测按键是否还是按下状态
		    {
			 Key_Change=1;					 //有按键按下使能变量,(此变量为1时才会处理键值数据)
			 Key_Con=0;						//变量清零,等待下次有按键按下
			}
		   break;
		  }
  }
 
 switch (Xintiao_Con)//此处与上面按键的检测类似
  {
   case 0:			 //默认Xintiao_Con是为0的
          {
		   if(!Xintiao)//每10ms(上面的定时器)检测一次脉搏是否有信号
		    {
			 Xintiao_Con++;//如果有信号,变量加一,程序就会往下走了
			}
		   break;
		  }
   case 1:
          {
		   if(!Xintiao)	   //每过10ms检测一下信号是否还存在
		    {
			 Xintiao_Con++;//存在就加一
			}
		   else
		    {
			 Xintiao_Con=0;//如果不存在了,检测时间很短,说明检测到的不是脉搏信号,可能是其他干扰,将变量清零,跳出此次检测
			} 
		   break;
		  }
  case 2:
          {
		   if(!Xintiao)
		    {
			 Xintiao_Con++;//存在就加一
			}
		   else
		    {
			 Xintiao_Con=0;//如果不存在了,检测时间很短,说明检测到的不是脉搏信号,可能是其他干扰,将变量清零,跳出此次检测
			} 
		   break;
		  }
  case 3:
          {
		   if(!Xintiao)
		    {
			 Xintiao_Con++;//存在就加一
			}
		   else
		    {
			 Xintiao_Con=0;//如果不存在了,检测时间很短,说明检测到的不是脉搏信号,可能是其他干扰,将变量清零,跳出此次检测
			} 
		   break;
		  }
  case 4:
          {
		   if(Xintiao)//超过30ms一直有信号,判定此次是脉搏信号,执行以下程序
		    {
			 if(Xintiao_Change==1)//心率计原理为检测两次脉冲间隔时间计算心率,变量Xintiao_Change第一次脉冲时为0的,所有走下面的else,第二次走这里
			  {
			   View_Data[0]=(60000/Xintiao_Jishu)/100+0x30;
	           View_Data[1]=(60000/Xintiao_Jishu)%100/10+0x30;
	           View_Data[2]=(60000/Xintiao_Jishu)%10+0x30;

			   if(((60000/Xintiao_Jishu)>=Xintiao_H)||((60000/Xintiao_Jishu)<=Xintiao_L))//心率不在范围内报警
			    speaker=0;			//蜂鸣器响
			   else
			    speaker=1;			//不响

			   View_Change=1;	   //计算出心率后启动显示
			   Xintiao_Jishu=0;	   //心跳计数清零
			   Xintiao_Change=0;   //计算出心率后该变量清零,准备下次检测心率
			   stop=0;			   //计算出心率后stop清零
			  }
			 else//第一次脉冲时Xintiao_Change为0
			  {
			   Xintiao_Jishu=0;	//脉冲计时变量清零,开始计时
			   Xintiao_Change=1;//Xintiao_Change置1,准备第二次检测到脉冲时计算心率
			  }
			 Xintiao_Con=0;	//清零,准备检测下一次脉冲
			 break;
			}
		  }
  }
}
/**定时器T0工作函数**/
void Time0() interrupt 1
{
 TH0=0xfc;		   //1ms
 TL0=0x18;		   //重新赋初值
 Xintiao_Jishu++;  //心跳计数加
 if(Xintiao_Jishu==5000)//心跳计数大于5000
  {
   Xintiao_Jishu=0;		//数据清零
   View_Change=1;		//显示位置1
   Xintiao_Change=0;	//置零,准备再次检测
   stop=1;	   //心跳计数超过5000后说明心率不正常或者没有测出,stop置1
   speaker=1;  //关闭蜂鸣器
  }
}
/**定时器初始化函数**/
void Tim_Init()
{
 EA=1;			  //打开中断总开关
 ET0=1;			  //打开T0中断允许开关
 ET1=1;			  //打开T1中断允许开关
 TMOD=0x11;		  //设定定时器状态
 TH0=0xfc;		   //1ms
 TL0=0x18;		   //赋初值
 
 TH1=0xd8;		   //10ms
 TL1=0xf0;		   //赋初值
}
/**在指定地址显示指定数量的指定字符**/
/**Adress_Com显示地址,Num_Adat显示字符数量,Adress_Data显示字符串内容**/ 
void lcd_1602_word(uchar Adress_Com,uchar Num_Adat,uchar *Adress_Data)
{
 uchar a=0;
 uchar Data_Word;
 LCD_WriteCom(Adress_Com); //选中地址
 for(a=0;a<Num_Adat;a++)   //for循环决定显示字符个数
  {
   Data_Word=*Adress_Data;	  //读取字符串数据
   LCD_WriteData(Data_Word);  //显示字符串
   Adress_Data++;			  //显示地址加一
  }
}

硬件设计

使用元器件:

单片机:STC89C51;

(注意:单片机是通用的,无论51还是52、无论stc还是at都一样,引脚功能都一样。程序也是一样的。)

9*15万用板;40脚IC座

8脚座;LM358芯片;

1602液晶;16p插针;

16p单排母座;按键*4;

5mm红外接收管;

5mm红外发射管;

103可调电位器;

2.2k电阻;1k电阻;

220Ω电阻;30k电阻*2;

200k电阻;10k电阻*6;

1.5k电阻;103排阻;

9012三极管;有源蜂鸣器;

1uf电容*2;331独石电容;

10uf电容;30pf电容*2;

12m晶振;Led(5MM红色);

DC电源接口;自锁开关;

导线:若干;

设计资料

01仿真图

本设计使用proteus8.9版本设计!具体如图!

添加图片注释,不超过 140 字(可选)

02原理图

本系统原理图采用Altium Designer19设计,具体如图!

添加图片注释,不超过 140 字(可选)

03程序

本设计使用软件keil5版本编程设计!具体如图!

添加图片注释,不超过 140 字(可选)

04设计报告

一万字参考报告,仅供参考!

添加图片注释,不超过 140 字(可选)

05设计资料

        资料获取请关注同名公众号,全部资料包括仿真源文件 、AD原理图、程序(含注释)、任务书、开题报告、设计报告、实物图、元件清单、演示视频等。具体内容如下,全网最全! !

可以关注下方公众号!

点赞分享一起学习成长。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/719679.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

带你走进CCS光源——环形低角度光源LDR2-LA系列

机器视觉系统中&#xff0c;光源起着重要作用&#xff0c;不同类型的光源应用也不同&#xff0c;选择合适的光源成像效果非常明显。今天我们一起来看看CCS光源——工业用环形低角度光源LDR2-LA系列。 LDR2-LA系列 采用柔性基板&#xff0c;创造最佳倾斜角度。 通过从低角度向…

微信小程序 - 出于性能原因,对长行跳过令牌化。长行的长度可通过 “editor.maxTokenizationLineLength” 进行配置

问题描述 出于性能原因&#xff0c;对长行跳过令牌化。长行的长度可通过 “editor.maxTokenizationLineLength” 进行配置。 解决方案 设置 - 编辑器设置 - 更多编辑器设置... 搜索&#xff1a;maxtoken&#xff0c;原来是 20000&#xff0c;我改成了 200000 即可~

海南云亿商务咨询有限公司抖店开店怎么样?

在数字化浪潮席卷全球的今天&#xff0c;电商行业日新月异&#xff0c;其中抖音电商以其独特的短视频直播模式&#xff0c;迅速崛起成为电商领域的新贵。海南云亿商务咨询有限公司&#xff0c;作为抖音电商服务的佼佼者&#xff0c;凭借专业的团队和丰富的经验&#xff0c;致力…

批量导出兜底回复对话,迭代优化聊天机器人 | Chatopera 云服务

持续优化知识库 聊天机器人的知识库&#xff0c;对话技能&#xff0c;需要长期的优化。这是因为&#xff0c;一方面&#xff0c;初期上线的机器人所依赖的数据量通常有限&#xff1b;另一方面&#xff0c;市场不断变化&#xff0c;客户产品新的问题。 上线聊天机器人的目的之…

基于Django、Bootstrap的电影推荐系统,算法基于用户的协同过滤算法,有爬虫有可视化后台

背景 基于Django和Bootstrap的电影推荐系统结合了用户协同过滤算法&#xff0c;通过爬虫技术获取电影数据&#xff0c;并在可视化后台展示推荐结果。该系统旨在提供个性化的电影推荐服务&#xff0c;帮助用户发现符合其喜好的电影。 用户协同过滤算法是一种常用的推荐算法&am…

蓝卓创始人褚健:工厂操作系统+APP,加速工业数字化转型

如何让众多的中小企业通过低成本的方式实现收益&#xff0c;享受到工业互联网、数字化转型带来的效益&#xff0c;是解决中小企业数字化转型难的核心问题。 中小企业规模庞大&#xff0c;数字化转型压力巨大 褚健表示&#xff0c;中国拥有最庞大的工业企业集群&#xff0c;全国…

STM32高级控制定时器(STM32F103):PWM输出模式

目录 概述 1 PWM模式介绍 2 PWM类型 2.1 PWM边缘对齐模式 2.2 PWM中心对齐模式 3 使用STM32Cube配置PWM 3.1 STM32Cube配置参数 3.2 生成Project 4 设置PWM占空比 4.1 函数介绍 4.3 函数源码 5 测试代码 5.1 编写测试代码 5.2 函数源码 6 运行代码 概述 本文主…

29. 透镜阵列

导论&#xff1a; 物理传播光学&#xff08;POP&#xff09;不仅可以用于简单系统&#xff0c;也可以设计优化复杂的光学系统&#xff0c;比如透镜阵列。 设计流程&#xff1a; 透镜阵列建模 在孔径类型中选择“入瞳直径”&#xff0c;并输入2 在视场设定中。设置一个视场&…

将自己md文件发布到自己的博客园实现文件的持久化存储

上传markdown文件到博客园 目录 【0】需求原因【1】功能【2】环境【最佳实践测试】 &#xff08;1&#xff09;查看 Typora 设置&#xff08;2&#xff09;配置 pycnblog 配置文件 config.yaml&#xff08;3&#xff09;运行 pycnblog 中的文件 cnblog_markdown.cmd&#xff0…

win11电脑桌面设置倒计时提醒教程

在日常工作中&#xff0c;我们经常需要处理大量的工作任务&#xff0c;而且很多任务都有时间限制。如果将这些任务记录在桌面上&#xff0c;并设置倒计时提醒&#xff0c;无疑会大大提高我们的工作效率。想象一下&#xff0c;在繁忙的工作间隙&#xff0c;你只需一瞥桌面&#…

如何经营好中医诊所?方法有哪些

在当今竞争激烈的医疗市场中&#xff0c;要想成功经营一家中医诊所&#xff0c;并不仅仅是提供传统的医疗服务&#xff0c;更需要与时俱进的战略思维和精细化的管理。过去被动获客、低效管理的模式已经不再适用&#xff0c;而如何拓展客源、提升服务质量、优化业务模式成为了中…

图书管理系统代码(Java)

1、运行演示 QQ2024528-205028-HD 详细讲解在这篇博客&#xff1a;JavaSE&#xff1a;图书管理系统-CSDN博客 2、所建的包 3、Java代码 3.1 book包 3.1.1 Book类代码 package book;/*** Created with IntelliJ IDEA.* Description:* User: dings* Date: 2024-05-13* Time:…

linux驱动学习(八)之内核定制与裁剪

一、内核的配置 1) 把相关硬件平台的配置文件拷贝给.config 2) 执行make menuconfig命令 关于内核配置说明:Arrow keys navigate the menu. 方向键对菜单有效<Enter> selects submenus --->. 如果有该符号"--->",则按Enter表示进入子菜单Highlighted …

【Ambari】Python调用Rest API 获取集群状态信息并发送钉钉告警

&#x1f341; 博主 "开着拖拉机回家"带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——&#x1f390;开着拖拉机回家_大数据运维-CSDN博客 &#x1f390;✨&#x1f341; &#x1fa81;&#x1f341; 希望本文能够给您带来一定的帮助&#x1f338;文…

路由控制和策略路由

文章目录 一、路由控制&#xff08;1&#xff09;、前言1.1.1-路由策略 &#xff08;2&#xff09;、正反掩码和通配符1.2.1-通配符 &#xff08;3&#xff09;、ACL1.3.1-ACL步长1.3.2-步长的作用1.3.3-TCP/UDP端口号 实验1:实验2: 二、前缀列表实验1:2.1.1-前缀列表的表达式2…

全功能知识付费小程序源码系统 界面支持万能DIY装修 带完整的安装代码包以及搭建部署教程

系统概述 在当今数字化时代&#xff0c;知识付费已经成为一种重要的商业模式。为了满足市场对于便捷、高效、个性化的知识付费解决方案的需求&#xff0c;小编给大家分享一款全功能知识付费小程序源码系统。这一系统不仅具备界面支持万能 DIY 装修的独特优势&#xff0c;还配备…

NetSarang Xshell Xftp v7 解锁版 (SSH远程终端工具)

前言 Xshell 7是一款SSH远程终端工具&#xff0c;轻松管理远程服务器&#xff0c;会话管理器&#xff0c;支持多选项卡管理主机&#xff0c;支持远程协议Telnet、Rlogin、SSH/SSH PKCS&#xff03;11、SFTP、Serial&#xff0c;具有Unicode编码支持、动态端口转发、自定义键盘…

【机器学习】——【线性回归模型】——详细【学习路线】

目录 1. 引言 2. 线性回归理论基础 2.1 线性模型概述 2.2 最小二乘法 3. 数学基础 3.1 矩阵运算 3.2 微积分 3.3 统计学 4. 实现与应用 4.1 使用Scikit-learn实现线性回归 4.2 模型评估 5. 深入理解 5.1 多元线性回归 5.2 特征选择 5.3 理解模型内部 6. 实战与项…

MacOS - 3 招快速去除桌面上的图标文件

在平时用 Mac 电脑的时候&#xff0c;会产生许多我们不用的或废弃的图标、文件&#xff0c;在 Mac 桌面上显得很乱&#xff0c;不仅影响美观也直接影响了我们工作的心情。下面我们分享 3 招快速去除桌面上的图标或文件的方法&#xff0c;有需要的朋友可以试一试。 1. 右键删除&…

Linux常用操作大全(上)

Linux常用操作 文章目录 Linux常用操作一、各类小技巧**1.ctrl c 强制停止****2.ctrl d 退出或登出**3.历史命令搜索4.光标移动快捷键5.清屏6.复制Ctrlshiftc7.粘贴Ctrlshiftv 二、软件安装1.概念2.yum与apt 三、systemctl控制服务四、软链接ln五、日期时区1.date查看日期2.修…