找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 3608|回復(fù): 1
打印 上一主題 下一主題
收起左側(cè)

51單片機(jī)心率計(jì)脈搏計(jì)仿真程序

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
51單片機(jī)心率計(jì)


單片機(jī)源程序如下:
  1. #include <reg52.h>
  2. #include <intrins.h>         //                         包含頭文件

  3. #define uint            unsigned int
  4. #define uchar           unsigned char
  5. #define ulong           unsigned long         //宏定義
  6. #define LCD_DATA        P0                                 //定義P0口為L(zhǎng)CD_DATA

  7. sbit LCD_RS =P2^5;
  8. sbit LCD_RW =P2^6;
  9. sbit LCD_E  =P2^7;                                                 //定義LCD控制引腳

  10. sbit Xintiao =P1^0 ;                                         //脈搏檢測(cè)輸入端定義
  11. sbit speaker =P2^4;                                                 //蜂鳴器引腳定義

  12. void delay5ms(void);   //誤差 0us
  13. void LCD_WriteData(uchar LCD_1602_DATA);         /********LCD1602數(shù)據(jù)寫入***********/
  14. void LCD_WriteCom(uchar LCD_1602_COM);                 /********LCD1602命令寫入***********/
  15. void lcd_1602_word(uchar Adress_Com,uchar Num_Adat,uchar *Adress_Data); /*1602字符顯示函數(shù),變量依次為字符顯示首地址,顯示字符長(zhǎng)度,所顯示的字符*/
  16. void InitLcd();//液晶初始化函數(shù)

  17. void Tim_Init();

  18. uchar Xintiao_Change=0;           //
  19. uint  Xintiao_Jishu;
  20. uchar stop;
  21. uchar View_Data[3];
  22. uchar View_L[3];
  23. uchar View_H[3];
  24. uchar Xintiao_H=100;        //脈搏上限
  25. uchar Xintiao_L=40;                //脈搏下限


  26. uchar Key_Change;
  27. uchar Key_Value;                //按鍵鍵值
  28. uchar View_Con;                        //設(shè)置的位(0正常工作,1設(shè)置上限,2設(shè)置下限)
  29. uchar View_Change;

  30. void main()          //主函數(shù)
  31. {
  32. InitLcd();
  33. Tim_Init();
  34. lcd_1602_word(0x80,16,"Heart Rate:     ");          //初始化顯示
  35. TR0=1;
  36. TR1=1;                                  //打開定時(shí)器
  37. while(1)                          //進(jìn)入循環(huán)
  38.   {
  39.    if(Key_Change)          //有按鍵按下并已經(jīng)得出鍵值
  40.     {
  41.          Key_Change=0;          //將按鍵使能變量清零,等待下次按鍵按下
  42.          View_Change=1;
  43.          switch(Key_Value)                                //判斷鍵值
  44.           {
  45.            case 1:                                                //設(shè)置鍵按下
  46.                  {
  47.                           View_Con++;                        //設(shè)置的位加
  48.                           if(View_Con==3)                //都設(shè)置好后將此變量清零
  49.                            View_Con=0;
  50.                           break;                                //跳出,下同
  51.                          }
  52.            case 2:                                                //加鍵按下
  53.                  {
  54.                           if(View_Con==2)                //判斷是設(shè)置上限
  55.                            {
  56.                                    if(Xintiao_H<150)        //上限數(shù)值小于150
  57.                                  Xintiao_H++;                //上限+
  58.                            }
  59.                           if(View_Con==1)                //如果是設(shè)置下限
  60.                            {
  61.                                    if(Xintiao_L<Xintiao_H-1)//下限值小于上限-1(下限值不能超過上限)
  62.                                  Xintiao_L++;                //下限值加
  63.                            }
  64.                           break;
  65.                          }
  66.            case 3:                                                //減鍵按下
  67.                  {
  68.                           if(View_Con==2)                //設(shè)置上限
  69.                            {
  70.                                    if(Xintiao_H>Xintiao_L+1)//上限數(shù)據(jù)大于下限+1(同樣上限值不能小于下限)
  71.                                  Xintiao_H--;                //上限數(shù)據(jù)減
  72.                            }
  73.                           if(View_Con==1)                //設(shè)置下限
  74.                            {
  75.                                    if(Xintiao_L>30)        //下限數(shù)據(jù)大于30時(shí)
  76.                                  Xintiao_L--;                //下限數(shù)據(jù)減
  77.                            }
  78.                           break;
  79.                          }
  80.          }
  81.         }
  82.    if(View_Change)//開始顯示變量
  83.     {
  84.          View_Change=0;//變量清零
  85.          if(stop==0)                          //心率正常時(shí)
  86.           {
  87.            if(View_Data[0]==0x30) //最高位為0時(shí)不顯示
  88.             View_Data[0]=' ';
  89.           }
  90.          else                                          //心率不正常(計(jì)數(shù)超過5000,也就是兩次信號(hào)時(shí)間超過5s)不顯示數(shù)據(jù)
  91.           {
  92.            View_Data[0]=' ';
  93.            View_Data[1]=' ';
  94.            View_Data[2]=' ';
  95.           }

  96.          switch(View_Con)
  97.           {
  98.            case 0: //正常顯示
  99.                   {
  100.                            lcd_1602_word(0x80,16,"Heart Rate:     ");//顯示一行數(shù)據(jù)
  101.                            lcd_1602_word(0xc0,16,"                ");//顯示第二行數(shù)據(jù)
  102.                            lcd_1602_word(0xcd,3,View_Data);                         //第二行顯示心率
  103.                            break;
  104.                           }
  105.            case 1: //設(shè)置下限時(shí)顯示
  106.                   {
  107.                            lcd_1602_word(0x80,16,"Heart Rate:     ");//第一行顯示心率
  108.                            lcd_1602_word(0x8d,3,View_Data);
  109.                           
  110.                            View_L[0]=Xintiao_L/100+0x30;                //將下限數(shù)據(jù)拆字
  111.                            View_L[1]=Xintiao_L%100/10+0x30;
  112.                            View_L[2]=Xintiao_L%10+0x30;

  113.                            if(View_L[0]==0x30)                                        //最高位為0時(shí),不顯示
  114.                             View_L[0]=' ';
  115.                           
  116.                            lcd_1602_word(0xC0,16,"Warning L :     ");//第二行顯示下限數(shù)據(jù)
  117.                            lcd_1602_word(0xCd,3,View_L);
  118.                            break;
  119.                           }
  120.            case 2: //設(shè)置上限時(shí)顯示(同上)
  121.                   {
  122.                            lcd_1602_word(0x80,16,"Heart Rate:     ");
  123.                            lcd_1602_word(0x8d,3,View_Data);
  124.                           
  125.                            View_H[0]=Xintiao_H/100+0x30;
  126.                            View_H[1]=Xintiao_H%100/10+0x30;
  127.                            View_H[2]=Xintiao_H%10+0x30;

  128.                            if(View_H[0]==0x30)
  129.                             View_H[0]=' ';
  130.                           
  131.                            lcd_1602_word(0xC0,16,"Warning H :     ");
  132.                            lcd_1602_word(0xCd,3,View_H);
  133.                            break;
  134.                           }
  135.           }
  136.         }
  137.   }
  138. }

  139. void Time1() interrupt 3                //定時(shí)器1服務(wù)函數(shù)
  140. {
  141.         static uchar Key_Con,Xintiao_Con;
  142.         TH1=0xd8;                   //10ms
  143.         TL1=0xf0;                   //重新賦初值
  144.         switch(Key_Con)   //無(wú)按鍵按下時(shí)此值為0
  145.         {
  146.                 case 0:                   //每10ms掃描此處
  147.                 {
  148.                         if((P3&0x07)!=0x07)//掃描按鍵是否有按下
  149.                         {
  150.                                 Key_Con++;                  //有按下此值加1,值為1
  151.                         }
  152.                         break;
  153.                 }
  154.                 case 1:                                          //10ms后二次進(jìn)入中斷后掃描此處(Key_Con為1)
  155.                 {
  156.                         if((P3&0x07)!=0x07)//第二次進(jìn)入中斷時(shí),按鍵仍然是按下(起到按鍵延時(shí)去抖的作用)
  157.                         {
  158.                                 Key_Con++;                  //變量加1,值為2
  159.                                 switch(P3&0x07)  //判斷是哪個(gè)按鍵按下
  160.                                 {
  161.                                         case 0x06:Key_Value=1;break;         //判斷好按鍵后將鍵值賦值給變量Key_Value
  162.                                         case 0x05:Key_Value=2;break;
  163.                                         case 0x03:Key_Value=3;break;
  164.                                 }
  165.                         }
  166.                         else                                                                 //如果10ms時(shí)沒有檢測(cè)到按鍵按下(按下時(shí)間過短)
  167.                         {
  168.                                 Key_Con=0;                                                 //變量清零,重新檢測(cè)按鍵
  169.                         }
  170.                         break;
  171.                 }
  172.                 case 2:                                                                         //20ms后檢測(cè)按鍵
  173.                 {
  174.                         if((P3&0x07)==0x07)                                 //檢測(cè)按鍵是否還是按下狀態(tài)
  175.                         {
  176.                                 Key_Change=1;                                         //有按鍵按下使能變量,(此變量為1時(shí)才會(huì)處理鍵值數(shù)據(jù))
  177.                                 Key_Con=0;                                                //變量清零,等待下次有按鍵按下
  178.                         }
  179.                         break;
  180.                 }
  181.         }
  182.        
  183.         switch (Xintiao_Con)//此處與上面按鍵的檢測(cè)類似
  184.         {
  185.                 case 0:                         //默認(rèn)Xintiao_Con是為0的
  186.                 {
  187.                         if(!Xintiao)//每10ms(上面的定時(shí)器)檢測(cè)一次脈搏是否有信號(hào)
  188.                         {
  189.                                 Xintiao_Con++;//如果有信號(hào),變量加一,程序就會(huì)往下走了
  190.                         }
  191.                         break;
  192.                 }
  193.                 case 1:
  194.                 {
  195.                         if(!Xintiao)           //每過10ms檢測(cè)一下信號(hào)是否還存在
  196.                         {
  197.                                 Xintiao_Con++;//存在就加一
  198.                         }
  199.                         else
  200.                         {
  201.                                 Xintiao_Con=0;//如果不存在了,檢測(cè)時(shí)間很短,說明檢測(cè)到的不是脈搏信號(hào),可能是其他干擾,將變量清零,跳出此次檢測(cè)
  202.                         }
  203.                         break;
  204.                 }
  205.                 case 2:
  206.                 {
  207.                         if(!Xintiao)
  208.                         {
  209.                                 Xintiao_Con++;//存在就加一
  210.                         }
  211.                         else
  212.                         {
  213.                                 Xintiao_Con=0;//如果不存在了,檢測(cè)時(shí)間很短,說明檢測(cè)到的不是脈搏信號(hào),可能是其他干擾,將變量清零,跳出此次檢測(cè)
  214.                         }
  215.                         break;
  216.                 }
  217.                 case 3:
  218.                 {
  219.                         if(!Xintiao)
  220.                         {
  221.                                 Xintiao_Con++;//存在就加一
  222.                         }
  223.                         else
  224.                         {
  225.                                 Xintiao_Con=0;//如果不存在了,檢測(cè)時(shí)間很短,說明檢測(cè)到的不是脈搏信號(hào),可能是其他干擾,將變量清零,跳出此次檢測(cè)
  226.                         }
  227.                         break;
  228.                 }
  229.                 case 4:
  230.                 {
  231.                         if(Xintiao)//超過30ms有信號(hào),判定此次是脈搏信號(hào),然后當(dāng)信號(hào)消失后,執(zhí)行以下程序
  232.                         {
  233.                                 if(Xintiao_Change==1)//心率計(jì)原理為檢測(cè)兩次脈沖間隔時(shí)間計(jì)算心率,變量Xintiao_Change第一次脈沖時(shí)為0的,所有走下面的else,第二次走這里
  234.                                 {
  235.                                         View_Data[0]=(60000/Xintiao_Jishu)/100+0x30;                  //計(jì)算心跳并拆字顯示:心跳計(jì)時(shí)是以1ms為單位,兩次心跳中間計(jì)數(shù)如果是1000次,也就是1000*1ms=1000ms=1s
  236.                                         View_Data[1]=(60000/Xintiao_Jishu)%100/10+0x30;          //那么計(jì)算出的一分鐘(60s)心跳數(shù)就是:60*1000/(1000*1ms)=60次          其中60是一分鐘60s,1000是一秒有1000ms,1000是計(jì)數(shù)值,1是一次計(jì)數(shù)對(duì)應(yīng) 的時(shí)間是1ms
  237.                                         View_Data[2]=(60000/Xintiao_Jishu)%10+0x30;                  //計(jì)算出的心跳數(shù)/100得到心跳的百位,%100是取余的,就是除以100的余數(shù),再除以10就得到十位了,以此類推
  238.                                                                                                   //拆字后的單個(gè)數(shù)據(jù)+0x30的目的是得到對(duì)應(yīng)數(shù)字的液晶顯示碼,數(shù)字0對(duì)應(yīng)的液晶顯示碼是0x30,1是0x30+1,以此類推
  239.                                         if(((60000/Xintiao_Jishu)>=Xintiao_H)||((60000/Xintiao_Jishu)<=Xintiao_L))//心率不在范圍內(nèi)報(bào)警
  240.                                         speaker=0;                        //蜂鳴器響
  241.                                         else
  242.                                         speaker=1;                        //不響
  243.                                        
  244.                                         View_Change=1;           //計(jì)算出心率后啟動(dòng)顯示
  245.                                         Xintiao_Jishu=0;           //心跳計(jì)數(shù)清零
  246.                                         Xintiao_Change=0;   //計(jì)算出心率后該變量清零,準(zhǔn)備下次檢測(cè)心率
  247.                                         stop=0;                           //計(jì)算出心率后stop清零
  248.                                 }
  249.                                 else//第一次脈沖時(shí)Xintiao_Change為0
  250.                                 {
  251.                                         Xintiao_Jishu=0;        //脈沖計(jì)時(shí)變量清零,開始計(jì)時(shí)
  252.                                         Xintiao_Change=1;//Xintiao_Change置1,準(zhǔn)備第二次檢測(cè)到脈沖時(shí)計(jì)算心率
  253.                                 }
  254.                                 Xintiao_Con=0;        //清零,準(zhǔn)備檢測(cè)下一次脈沖
  255.                                 break;
  256.                         }
  257.                 }
  258.         }
  259. }
  260. /**定時(shí)器T0工作函數(shù)**/
  261. void Time0() interrupt 1
  262. {
  263. TH0=0xfc;                   //1ms
  264. TL0=0x18;                   //重新賦初值
  265. Xintiao_Jishu++;  //心跳計(jì)數(shù)加
  266. if(Xintiao_Jishu==5000)//心跳計(jì)數(shù)大于5000
  267.   {
  268.    Xintiao_Jishu=0;                //數(shù)據(jù)清零
  269.    View_Change=1;                //顯示位置1
  270.    Xintiao_Change=0;        //置零,準(zhǔn)備再次檢測(cè)
  271.    stop=1;           //心跳計(jì)數(shù)超過5000后說明心率不正;蛘邲]有測(cè)出,stop置1
  272.    speaker=1;  //關(guān)閉蜂鳴器
  273.   }
  274. }
  275. /**定時(shí)器初始化函數(shù)**/
  276. void Tim_Init()
  277. {
  278. EA=1;                          //打開中斷總開關(guān)
  279. ET0=1;                          //打開T0中斷允許開關(guān)
  280. ET1=1;                          //打開T1中斷允許開關(guān)
  281. TMOD=0x11;                  //設(shè)定定時(shí)器狀態(tài)
  282. TH0=0xfc;                   //1ms
  283. TL0=0x18;                   //賦初值

  284. TH1=0xd8;                   //10ms
  285. TL1=0xf0;                   //賦初值
  286. }
  287. /**在指定地址顯示指定數(shù)量的指定字符**/
  288. /**Adress_Com顯示地址,Num_Adat顯示字符數(shù)量,Adress_Data顯示字符串內(nèi)容**/
  289. void lcd_1602_word(uchar Adress_Com,uchar Num_Adat,uchar *Adress_Data)
  290. {
  291. uchar a=0;
  292. uchar Data_Word;
  293. LCD_WriteCom(Adress_Com); //選中地址
  294. for(a=0;a<Num_Adat;a++)   //for循環(huán)決定顯示字符個(gè)數(shù)
  295.   {
  296.    Data_Word=*Adress_Data;          //讀取字符串?dāng)?shù)據(jù)
  297.    LCD_WriteData(Data_Word);  //顯示字符串
  298.    Adress_Data++;                          //顯示地址加一
  299.   }
  300. }

  301. /***************1602函數(shù)*******************/
  302. void LCD_WriteData(uchar LCD_1602_DATA)         /********LCD1602數(shù)據(jù)寫入***********/
  303. {
  304. delay5ms();  //操作前短暫延時(shí),保證信號(hào)穩(wěn)定
  305. LCD_E=0;
  306. LCD_RS=1;
  307. LCD_RW=0;
  308. _nop_();
  309. LCD_E=1;
  310. LCD_DATA=LCD_1602_DATA;
  311. LCD_E=0;
  312. LCD_RS=0;
  313. }

  314. /********LCD1602命令寫入***********/
  315. void LCD_WriteCom(uchar LCD_1602_COM)
  316. {
  317. delay5ms();//操作前短暫延時(shí),保證信號(hào)穩(wěn)定
  318. LCD_E=0;
  319. LCD_RS=0;
  320. LCD_RW=0;
  321. _nop_();
  322. LCD_E=1;
  323. LCD_DATA=LCD_1602_COM;
  324. LCD_E=0;
  325. LCD_RS=0;
  326. }


  327. void InitLcd()                   //初始化液晶函數(shù)
  328. {
  329. delay5ms();
  330. delay5ms();
  331. LCD_WriteCom(0x38); //display mode
  332. LCD_WriteCom(0x38); //display mode
  333. LCD_WriteCom(0x38); //display mode
  334. LCD_WriteCom(0x06); //顯示光標(biāo)移動(dòng)位置
  335. LCD_WriteCom(0x0c); //顯示開及光標(biāo)設(shè)置
  336. LCD_WriteCom(0x01); //顯示清屏
  337. delay5ms();
  338. delay5ms();
  339. }

  340. void delay5ms(void)   //5ms延時(shí)函數(shù)
  341. {
  342.     unsigned char a,b;
  343.     for(b=185;b>0;b--)
  344.         for(a=12;a>0;a--);
  345. }
復(fù)制代碼

所有資料51hei提供下載:
51脈搏計(jì)仿真.zip (635.97 KB, 下載次數(shù): 78)


評(píng)分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎(jiǎng)勵(lì)!

查看全部評(píng)分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏3 分享淘帖 頂 踩
回復(fù)

使用道具 舉報(bào)

沙發(fā)
ID:1080928 發(fā)表于 2023-5-30 17:41 | 只看該作者
程序有問題吧。
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表