久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2050|回復: 4
收起左側

這個單片機超聲波測距程序怎么改

[復制鏈接]
ID:143767 發表于 2019-11-23 13:17 | 顯示全部樓層 |閱讀模式
這個超聲波測距報警程序原來報警時蜂鳴器是斷續鳴叫的,現在我想改成一直鳴叫,應該改哪里,怎樣改,謝謝!

完整的程序代碼如下:

  1. #include <reg52.h>                 //調用單片機頭文件
  2. #define uchar unsigned char  //無符號字符型 宏定義        變量范圍0~255
  3. #define uint  unsigned int         //無符號整型 宏定義        變量范圍0~65535
  4. #include <intrins.h>
  5. uchar a_a;


  6. //數碼管段選定義      0     1    2    3    4    5        6         7          8           9        
  7. uchar code smg_du[]={0x28,0xee,0x32,0xa2,0xe4,0xa1,0x21,0xea,0x20,0xa0,
  8.                                            0x60,0x25,0x39,0x26,0x31,0x71,0xff};         //斷碼


  9. uchar dis_smg[8]   ={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8};

  10. //數碼管位選定義
  11. sbit smg_we1 = P3^4;            //數碼管位選定義
  12. sbit smg_we2 = P3^5;
  13. sbit smg_we3 = P3^6;
  14. sbit smg_we4 = P3^7;

  15. sbit c_send   = P3^2;                //超聲波發射
  16. sbit c_recive = P3^3;                //超聲波接收

  17. sbit beep = P2^3;   //蜂鳴器IO口定義
  18. uchar smg_i = 3;    //顯示數碼管的個位數
  19. bit flag_300ms ;

  20. long distance;                //距離
  21. uint set_d;                    //距離
  22. uchar flag_csb_juli;    //超聲波超出量程
  23. uint  flag_time0;       //用來保存定時器0的時候的

  24. uchar menu_1;           //菜單設計的變量


  25. /***********************1ms延時函數*****************************/
  26. void delay_1ms(uint q)
  27. {
  28.         uint i,j;
  29.         for(i=0;i<q;i++)
  30.                 for(j=0;j<120;j++);
  31. }

  32. /***********************處理距離函數****************************/
  33. void smg_display()
  34. {
  35.         dis_smg[0] = smg_du[distance % 10];
  36.         dis_smg[1] = smg_du[distance / 10 % 10];
  37.         dis_smg[2] = smg_du[distance / 100 % 10] & 0xdf; ;        
  38. }


  39. #define RdCommand 0x01 //定義ISP的操作命令
  40. #define PrgCommand 0x02
  41. #define EraseCommand 0x03
  42. #define Error 1
  43. #define Ok 0
  44. #define WaitTime 0x01 //定義CPU的等待時間
  45. sfr ISP_DATA=0xe2;  //寄存器申明
  46. sfr ISP_ADDRH=0xe3;
  47. sfr ISP_ADDRL=0xe4;
  48. sfr ISP_CMD=0xe5;
  49. sfr ISP_TRIG=0xe6;
  50. sfr ISP_CONTR=0xe7;

  51. /* ================ 打開 ISP,IAP 功能 ================= */
  52. void ISP_IAP_enable(void)
  53. {
  54.          EA = 0;       /* 關中斷   */
  55.          ISP_CONTR = ISP_CONTR & 0x18;       /* 0001,1000 */
  56.          ISP_CONTR = ISP_CONTR | WaitTime; /* 寫入硬件延時 */
  57.          ISP_CONTR = ISP_CONTR | 0x80;       /* ISPEN=1  */
  58. }
  59. /* =============== 關閉 ISP,IAP 功能 ================== */
  60. void ISP_IAP_disable(void)
  61. {
  62.          ISP_CONTR = ISP_CONTR & 0x7f; /* ISPEN = 0 */
  63.          ISP_TRIG = 0x00;
  64.          EA   =   1;   /* 開中斷 */
  65. }
  66. /* ================ 公用的觸發代碼 ==================== */
  67. void ISPgoon(void)
  68. {
  69.          ISP_IAP_enable();   /* 打開 ISP,IAP 功能 */
  70.          ISP_TRIG = 0x46;  /* 觸發ISP_IAP命令字節1 */
  71.          ISP_TRIG = 0xb9;  /* 觸發ISP_IAP命令字節2 */
  72.          _nop_();
  73. }
  74. /* ==================== 字節讀 ======================== */
  75. unsigned char byte_read(unsigned int byte_addr)
  76. {
  77.         EA = 0;
  78.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);/* 地址賦值 */
  79.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  80.          ISP_CMD   = ISP_CMD & 0xf8;   /* 清除低3位  */
  81.          ISP_CMD   = ISP_CMD | RdCommand; /* 寫入讀命令 */
  82.          ISPgoon();       /* 觸發執行  */
  83.          ISP_IAP_disable();    /* 關閉ISP,IAP功能 */
  84.          EA  = 1;
  85.          return (ISP_DATA);    /* 返回讀到的數據 */
  86. }
  87. /* ================== 扇區擦除 ======================== */
  88. void SectorErase(unsigned int sector_addr)
  89. {
  90.          unsigned int iSectorAddr;
  91.          iSectorAddr = (sector_addr & 0xfe00); /* 取扇區地址 */
  92.          ISP_ADDRH = (unsigned char)(iSectorAddr >> 8);
  93.          ISP_ADDRL = 0x00;
  94.          ISP_CMD = ISP_CMD & 0xf8;   /* 清空低3位  */
  95.          ISP_CMD = ISP_CMD | EraseCommand; /* 擦除命令3  */
  96.          ISPgoon();       /* 觸發執行  */
  97.          ISP_IAP_disable();    /* 關閉ISP,IAP功能 */
  98. }
  99. /* ==================== 字節寫 ======================== */
  100. void byte_write(unsigned int byte_addr, unsigned char original_data)
  101. {
  102.          EA  = 0;
  103. //         SectorErase(byte_addr);
  104.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);  /* 取地址  */
  105.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  106.          ISP_CMD  = ISP_CMD & 0xf8;    /* 清低3位 */
  107.          ISP_CMD  = ISP_CMD | PrgCommand;  /* 寫命令2 */
  108.          ISP_DATA = original_data;   /* 寫入數據準備 */
  109.          ISPgoon();       /* 觸發執行  */
  110.          ISP_IAP_disable();     /* 關閉IAP功能 */
  111.          EA =1;
  112. }


  113. /******************把數據保存到單片機內部eeprom中******************/
  114. void write_eeprom()
  115. {
  116.         SectorErase(0x2000);
  117.         byte_write(0x2000, set_d % 256);
  118.         byte_write(0x2001, set_d / 256);
  119.         byte_write(0x2058, a_a);        
  120. }

  121. /******************把數據從單片機內部eeprom中讀出來*****************/
  122. void read_eeprom()
  123. {
  124.         set_d  = byte_read(0x2001);
  125.         set_d <<= 8;
  126.         set_d  |= byte_read(0x2000);
  127.         a_a      = byte_read(0x2058);
  128. }

  129. /**************開機自檢eeprom初始化*****************/
  130. void init_eeprom()
  131. {
  132.         read_eeprom();                //先讀
  133.         if(a_a != 1)                //新的單片機初始單片機內問eeprom
  134.         {
  135.                 set_d = 50;
  136.                 a_a = 1;
  137.                 write_eeprom();           //保存數據
  138.         }        
  139. }

  140. /********************獨立按鍵程序*****************/
  141. uchar key_can;         //按鍵值

  142. void key()         //獨立按鍵程序
  143. {
  144.         static uchar key_new;
  145.         key_can = 20;                   //按鍵值還原
  146.         P2 |= 0x07;
  147.         if((P2 & 0x07) != 0x07)                //按鍵按下
  148.         {
  149.                 delay_1ms(1);                     //按鍵消抖動
  150.                 if(((P2 & 0x07) != 0x07) && (key_new == 1))
  151.                 {                                                //確認是按鍵按下
  152.                         key_new = 0;
  153.                         switch(P2 & 0x07)
  154.                         {
  155.                                 case 0x06: key_can = 3; break;           //得到k2鍵值
  156.                                 case 0x05: key_can = 2; break;           //得到k3鍵值
  157.                                 case 0x03: key_can = 1; break;           //得到k4鍵值
  158.                         }
  159.                 }                        
  160.         }
  161.         else
  162.                 key_new = 1;        
  163. }

  164. /****************按鍵處理顯示函數***************/
  165. void key_with()
  166. {
  167.         if(key_can == 1)                //設置鍵
  168.         {
  169.                 menu_1 ++;
  170.                 if(menu_1 >= 2)
  171.                 {
  172.                         menu_1 = 0;
  173.                         smg_i = 3;                //只顯示3位數碼管
  174.                 }
  175.                 if(menu_1 == 1)
  176.                 {
  177.                         smg_i = 4;           //只顯示4位數碼管
  178.                 }
  179.         }
  180.         if(menu_1 == 1)                        //設置報警
  181.         {
  182.                 if(key_can == 2)
  183.                 {
  184.                         set_d ++ ;                //加1
  185.                         if(set_d > 400)
  186.                                 set_d = 400;
  187.                 }
  188.                 if(key_can == 3)
  189.                 {
  190.                         set_d -- ;                //減1
  191.                         if(set_d <= 1)
  192.                                 set_d = 1;
  193.                 }
  194.                 dis_smg[0] = smg_du[set_d % 10];                   //取小數顯示
  195.                 dis_smg[1] = smg_du[set_d / 10 % 10] ;         //取個位顯示
  196.                 dis_smg[2] = smg_du[set_d / 100 % 10] & 0xdf ; //取十位顯示
  197.                 dis_smg[3] = 0x60;                //a
  198.                 write_eeprom();                           //保存數據
  199.         }        
  200. }  

  201. /****************報警函數***************/
  202. void clock_h_l()
  203. {
  204.         static uchar value;
  205.         if(distance <= set_d)
  206.         {
  207.                 value ++;  //消除實際距離在設定距離左右變化時的干擾
  208.                 if(value >= 2)
  209.                 {
  210.                 beep =0;         
  211.                 //beep = ~beep;           //蜂鳴器報警
  212.                   
  213.         
  214.                 }
  215.         }
  216.         else
  217.         {
  218.                 value = 0;
  219.                 beep = 1;                //取消報警
  220.         }        
  221. }

  222. /***********************數碼位選函數*****************************/
  223. void smg_we_switch(uchar i)
  224. {
  225.         switch(i)
  226.         {
  227.                 case 0: smg_we1 = 0;  smg_we2 = 1; smg_we3 = 1;  smg_we4 = 1; break;
  228.                 case 1: smg_we1 = 1;  smg_we2 = 0; smg_we3 = 1;  smg_we4 = 1; break;
  229.                 case 2: smg_we1 = 1;  smg_we2 = 1; smg_we3 = 0;  smg_we4 = 1; break;
  230.                 case 3: smg_we1 = 1;  smg_we2 = 1; smg_we3 = 1;  smg_we4 = 0; break;
  231.         }        
  232. }

  233. /***********************數碼顯示函數*****************************/
  234. void display()
  235. {
  236.         static uchar i;   
  237.         i++;
  238.         if(i >= smg_i)
  239.                 i = 0;        
  240.         smg_we_switch(i);                 //位選
  241.         P1 = dis_smg[i];                 //段選               
  242. }

  243. /******************小延時函數*****************/
  244. void delay()
  245. {
  246.         _nop_();                            //執行一條_nop_()指令就是1us
  247.         _nop_();
  248.         _nop_();
  249.         _nop_();
  250.         _nop_();
  251.         _nop_();
  252.         _nop_();
  253.         _nop_();
  254.         _nop_();  
  255.         _nop_();
  256. }


  257. /*********************超聲波測距程序*****************************/
  258. void send_wave()
  259. {
  260.         c_send = 1;                           //10us的高電平觸發
  261.         delay();
  262.         c_send = 0;         
  263.         TH0 = 0;                          //給定時器0清零
  264.         TL0 = 0;
  265.         TR0 = 0;                                  //關定時器0定時
  266.         while(!c_recive);                  //當c_recive為零時等待
  267.         TR0=1;
  268.         while(c_recive)                      //當c_recive為1計數并等待
  269.         {
  270.                 flag_time0 = TH0 * 256 + TL0;
  271.                 if((flag_time0 > 40000))      //當超聲波超過測量范圍時,顯示3個888
  272.                 {
  273.                         TR0 = 0;
  274.                         flag_csb_juli = 2;
  275.                         distance = 888;
  276.                         break ;               
  277.                 }
  278.                 else
  279.                 {
  280.                         flag_csb_juli = 1;        
  281.                 }
  282.         }
  283.         if(flag_csb_juli == 1)
  284.         {        
  285.                 TR0=0;                                                         //關定時器0定時
  286.                 distance =flag_time0;                         //讀出定時器0的時間
  287.                 distance *= 0.017;               // 0.017 = 340M / 2 = 170M = 0.017M 算出來是米
  288.                 if((distance > 500))                                 //距離 = 速度 * 時間
  289.                 {        
  290.                         distance = 888;                                 //如果大于3.8m就超出超聲波的量程
  291.                 }
  292.         }  
  293. }


  294. /*********************定時器0、定時器1初始化******************/
  295. void time_init()         
  296. {
  297.         EA  = 1;                   //開總中斷
  298.         TMOD = 0X11;          //定時器0、定時器1工作方式1
  299.         ET0 = 0;                  //關定時器0中斷
  300.         TR0 = 1;                  //允許定時器0定時
  301.         ET1 = 1;                  //開定時器1中斷
  302.         TR1 = 1;                  //允許定時器1定時        
  303. }



  304. /***************主函數*****************/
  305. void main()
  306. {
  307.         beep = 0;                 //開機叫一聲   
  308.         delay_1ms(150);
  309.         P0 = P1 = P2 = P3 = 0xff;           //初始化單片機IO口為高電平
  310.         send_wave();        //測距離函數
  311.         smg_display();        //處理距離顯示函數
  312.         time_init();        //定時器初始化程序
  313.         init_eeprom();  //開始初始化保存的數據
  314.         send_wave();        //測距離函數
  315.         send_wave();        //測距離函數
  316.         while(1)
  317.         {                  
  318.                 if(flag_300ms == 1)
  319.                 {               
  320.                         flag_300ms = 0;
  321.                         clock_h_l();    //報警函數
  322.                         if(beep == 1)
  323.                                 send_wave();        //測距離函數
  324.                         if(menu_1 == 0)
  325.                                 smg_display();         //處理距離顯示函數
  326.                 }
  327.                 key();                                         //按鍵函數
  328.                 if(key_can < 10)
  329.                 {
  330.                         key_with();                         //按鍵處理函數
  331.                 }
  332.         }
  333. }

  334. /*********************定時器1中斷服務程序************************/
  335. void time1_int() interrupt 3
  336. {        
  337.         static uchar value;                         //定時2ms中斷一次
  338.         TH1 = 0xf8;
  339.         TL1 = 0x30;     //2ms
  340.         display();                //數碼管顯示函數
  341.         value++;
  342.         if(value >= 150)
  343.         {
  344.                 value = 0;
  345.                 flag_300ms = 1;
  346.         }
  347. }
復制代碼


回復

使用道具 舉報

ID:185698 發表于 2019-11-23 17:48 | 顯示全部樓層
//    if(beep == 1)
                                send_wave();        //測距離函數
回復

使用道具 舉報

ID:123289 發表于 2019-11-24 10:47 | 顯示全部樓層
重寫一個讓蜂鳴器長鳴的程序,其它功能全不要。
成功后,你就會改了。
回復

使用道具 舉報

ID:648095 發表于 2019-11-24 11:08 | 顯示全部樓層
             key_with();                         //按鍵處理函數
回復

使用道具 舉報

ID:629731 發表于 2019-11-24 13:01 | 顯示全部樓層
我覺得可以慢慢看。一條一條的排查
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 亚洲成人动漫在线观看 | 亚洲激情在线视频 | 黄色一级大片在线免费看产 | 久久精品国产一区二区三区不卡 | 国产真实精品久久二三区 | 在线免费小视频 | 精品久久久久久久久久久 | 亚洲女优在线播放 | 国产精品a久久久久 | 天天av网| 一区二区高清 | 午夜在线观看视频 | 日本福利视频免费观看 | 欧美日韩视频在线播放 | 美女一级黄| 在线电影日韩 | 欧美啪啪网站 | 中文字幕第二十页 | 国产一区二区三区四区在线观看 | 日韩精品在线播放 | 欧美精品v国产精品v日韩精品 | 国产高清一区二区三区 | 最新国产精品视频 | 一道本不卡视频 | 99久久免费精品国产男女高不卡 | 91免费版在线观看 | 欧美日韩一区二区三区四区 | www.婷婷 | 久久久这里都是精品 | 国产ts人妖系列高潮 | 四虎永久影院 | 国产精品视频网 | 久久精品成人 | 好好的日在线视频 | 久久国产综合 | 丁香久久 | 99色在线视频 | 午夜精品一区 | 91精品久久久久久久久 | 久久免费视频1 | 国产精品久久久久久妇女6080 |