找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2915|回复: 0
收起左侧

如何把超声波测距仪的蜂鸣器一段报警变成二段

[复制链接]
ID:266917 发表于 2017-12-26 10:29 | 显示全部楼层 |阅读模式
单片机源程序如下:
  1. #include <reg52.h>         //调用单片机头文件
  2. #define uchar unsigned char  //无符号字符型宏定义       变量范围0~255
  3. #define uint  unsigned int     //无符号整型 宏定义  变量范围0~65535
  4. #include <intrins.h>
  5. #include "eeprom52.h"


  6. //数码管段选定义      0     1   2    3    4   5  6       7      8     9
  7. uchar codesmg_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.         uinti,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. /******************把数据保存到单片机内部eeprom中******************/
  40. void write_eeprom()
  41. {
  42.         SectorErase(0x2000);
  43.         byte_write(0x2000,set_d % 256);
  44.         byte_write(0x2001,set_d / 256);
  45.         byte_write(0x2058,a_a);      
  46. }

  47. /******************把数据从单片机内部eeprom中读出来*****************/
  48. void read_eeprom()
  49. {
  50.         set_d  = byte_read(0x2001);
  51.         set_d<<= 8;
  52.         set_d  |= byte_read(0x2000);
  53.         a_a      = byte_read(0x2058);
  54. }

  55. /**************开机自检eeprom初始化*****************/
  56. void init_eeprom()
  57. {
  58.         read_eeprom();                //先读
  59.         if(a_a!= 1)              //新的单片机初始单片机内问eeprom
  60.         {
  61.                  set_d= 50;
  62.                  a_a= 1;
  63.                  write_eeprom();        //保存数据
  64.         }      
  65. }

  66. /********************独立按键程序*****************/
  67. uchar key_can;         //按键值

  68. void key()       //独立按键程序
  69. {
  70.         staticuchar key_new;
  71.         key_can= 20;                   //按键值还原
  72.         P2|= 0x07;
  73.         if((P2& 0x07) != 0x07)         //按键按下
  74.         {
  75.                  delay_1ms(1);             //按键消抖动
  76.                  if(((P2& 0x07) != 0x07) && (key_new == 1))
  77.                  {                                                //确认是按键按下
  78.                          key_new= 0;
  79.                          switch(P2& 0x07)
  80.                          {
  81.                                   case0x06: key_can = 3; break;         //得到k2键值
  82.                                   case0x05: key_can = 2; break;         //得到k3键值
  83.                                   case0x03: key_can = 1; break;         //得到k4键值
  84.                          }
  85.                  }                       
  86.         }
  87.         else
  88.                  key_new= 1;  
  89. }

  90. /****************按键处理显示函数***************/
  91. void key_with()
  92. {
  93.         if(key_can== 1)             //设置键
  94.         {
  95.                  menu_1++;
  96.                  if(menu_1>= 2)
  97.                  {
  98.                          menu_1= 0;
  99.                          smg_i= 3;                //只显示3位数码管
  100.                  }
  101.                  if(menu_1== 1)
  102.                  {
  103.                          smg_i= 4;          //只显示4位数码管
  104.                  }
  105.         }
  106.         if(menu_1== 1)                      //设置报警
  107.         {
  108.                  if(key_can== 2)
  109.                  {
  110.                          set_d++ ;                 //加1
  111.                          if(set_d> 400)
  112.                                   set_d= 400;
  113.                  }
  114.                  if(key_can== 3)
  115.                  {
  116.                          set_d-- ;          //减1
  117.                          if(set_d<= 1)
  118.                                   set_d= 1;
  119.                  }
  120.                  dis_smg[0]= smg_du[set_d % 10];                  //取小数显示
  121.                  dis_smg[1]= smg_du[set_d / 10 % 10] ;         //取个位显示
  122.                  dis_smg[2]= smg_du[set_d / 100 % 10] & 0xdf ; //取十位显示
  123.                  dis_smg[3]= 0x60;         //a
  124.                  write_eeprom();                          //保存数据
  125.         }      
  126. }  

  127. /****************报警函数***************/
  128. void clock_h_l()
  129. {
  130.         staticuchar value;
  131.         if(distance<= set_d)
  132.         {
  133.                  value++;  //消除实际距离在设定距离左右变化时的干扰
  134.                  if(value>= 2)
  135.                  {
  136.                        beep = ~beep;    //蜂鸣器报警 ~是位运算符,在二进制中是按位取反的意思
  137.                  }
  138.         }
  139.         else
  140.         {
  141.                  value= 0;
  142.                  beep= 1;         //取消报警
  143.         }      
  144. }

  145. /***********************数码位选函数*****************************/
  146. void smg_we_switch(uchar i)
  147. {
  148.         switch(i)
  149.         {
  150.                 case0: smg_we1 = 0;  smg_we2 = 1; smg_we3 =1;  smg_we4 = 1; break;
  151.                 case1: smg_we1 = 1;  smg_we2 = 0; smg_we3 =1;  smg_we4 = 1; break;
  152.                 case2: smg_we1 = 1;  smg_we2 = 1; smg_we3 =0;  smg_we4 = 1; break;
  153.                 case3: smg_we1 = 1;  smg_we2 = 1; smg_we3 =1;  smg_we4 = 0; break;
  154.         }      
  155. }

  156. /***********************数码显示函数*****************************/
  157. void display()
  158. {
  159.         staticuchar i;   
  160.         i++;
  161.         if(i>= smg_i)
  162.                  i= 0;        
  163.         smg_we_switch(i);           //位选
  164.         P1= dis_smg[ i];               //段选            
  165. }

  166. /******************小延时函数*****************/
  167. void delay()
  168. {
  169.         _nop_();                            //执行一条_nop_()指令就是1us
  170.         _nop_();
  171.         _nop_();
  172.         _nop_();
  173.         _nop_();
  174.         _nop_();
  175.         _nop_();
  176.         _nop_();
  177.         _nop_();  
  178.         _nop_();
  179. }


  180. /*********************超声波测距程序*****************************/
  181. void send_wave()
  182. {
  183.         c_send= 1;                         //10us的高电平触发
  184.         delay();
  185.         c_send= 0;      
  186.         TH0= 0;                    //给定时器0清零
  187.         TL0= 0;
  188.         TR0= 0;                             //关定时器0定时
  189.         while(!c_recive);              //当c_recive为零时等待
  190.         TR0=1;
  191.         while(c_recive)                      //当c_recive为1计数并等待
  192.         {
  193.                  flag_time0= TH0 * 256 + TL0;
  194.                  if((flag_time0> 40000))      //当超声波超过测量范围时,显示3个888
  195.                  {
  196.                          TR0= 0;
  197.                          flag_csb_juli= 2;
  198.                          distance= 888;
  199.                          break;              
  200.                  }
  201.                  else
  202.                  {
  203.                          flag_csb_juli= 1;   
  204.                  }
  205.         }
  206.         if(flag_csb_juli== 1)
  207.         {      
  208.                  TR0=0;                                                      //关定时器0定时
  209.                  distance=flag_time0;                       //读出定时器0的时间
  210.                  distance*= 0.017;               // 0.017 = 340M /2 = 170M = 0.017M 算出来是米
  211.                  if((distance> 500))                           //距离 = 速度 * 时间
  212.                  {      
  213.                          distance= 888;                                  //如果大于3.8m就超出超声波的量程
  214.                  }
  215.         }  
  216. }


  217. /*********************定时器0、定时器1初始化******************/
  218. void time_init()         
  219. {
  220.         EA  = 1;          //开总中断
  221.         TMOD= 0X11;        //定时器0、定时器1工作方式1
  222.         ET0= 0;            //关定时器0中断
  223.         TR0= 1;            //允许定时器0定时
  224.         ET1= 1;            //开定时器1中断
  225.         TR1= 1;            //允许定时器1定时  
  226. }



  227. /***************主函数*****************/
  228. void main()
  229. {
  230.         beep= 0;          //开机叫一声   
  231.         delay_1ms(150);
  232.         P0= P1 = P2 = P3 = 0xff;         //初始化单片机IO口为高电平
  233.         send_wave();  //测距离函数
  234.         smg_display();        //处理距离显示函数
  235.         time_init();      //定时器初始化程序
  236.         init_eeprom();  //开始初始化保存的数据
  237.         send_wave();  //测距离函数
  238.         send_wave();  //测距离函数
  239.         while(1)
  240.         {               
  241.                  if(flag_300ms== 1)
  242.                  {               
  243.                          flag_300ms= 0;
  244.                          clock_h_l();    //报警函数
  245.                          if(beep== 1)
  246.                                   send_wave();  //测距离函数
  247.                          if(menu_1== 0)
  248.                                   smg_display();         //处理距离显示函数
  249.                  }
  250.                  key();                                         //按键函数
  251.                  if(key_can< 10)
  252.                  {
  253.                          key_with();                       //按键处理函数
  254.                  }
  255.         }
  256. }

  257. /*********************定时器1中断服务程序************************/
  258. void time1_int() interrupt 3
  259. {      
  260.         staticuchar value;                     //定时2ms中断一次
  261.         TH1= 0xf8;
  262.         TL1= 0x30;     //2ms
  263.         display();         //数码管显示函数
  264.         value++;
  265.         if(value>= 150)
  266.         {
  267.                  value= 0;
  268.                  flag_300ms= 1;
  269.         }
  270. }

复制代码





回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表