找回密码
 立即注册

QQ登录

只需一步,快速开始

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

[原创]第八届蓝桥杯决赛:超声波测距机源程序

[复制链接]
ID:331814 发表于 2019-5-12 23:22 | 显示全部楼层 |阅读模式
本帖最后由 Amistad 于 2019-5-12 23:25 编辑

这次分享得是第八届的题目源码,我总结的一下第三届到底九届决赛考试考的器件,如下图所示 1.JPG ,大家可以看到板子上的lcd12864,1602,红外,在这几届里面一次没考,但是不代表以后不会考,参加比赛的同学还是要看一下这三个器件,下面给出本次题目,与主文件程序代码: 第8届:超声波测距机试题题目_Page1.jpg 第8届:超声波测距机试题题目_Page2.jpg 第8届:超声波测距机试题题目_Page3.jpg

单片机源程序如下:
  1. #include "main.h"
  2. #include "Display.H"
  3. #include "stdio.H"
  4. #include "KEY.h"
  5. #include "TIME.h"
  6. #include "DELAY.h"
  7. #include "i2c.h"
  8. #include "CONFIG.H"
  9. #include "intrins.h"
  10. //发送40KHz脉冲延时函数
  11. #define somenop {_nop_();_nop_();_nop_();_nop_();_nop_();\
  12.                  _nop_();_nop_();_nop_();_nop_(); _nop_();}

  13. union flo_to_ch
  14. {
  15.         u16 f_num;
  16.         u8 u_num[2];//共用内存单元,u_num[2]分别为f32数据u16的2个字节
  17. };

  18. union flo_to_ch diyi_juli_h,dier_juli_h,disan_juli_h,disi_juli_h;//四次距离定义
  19. u16 juli=0,juli_s=0;//这次距离和上次距离
  20. u8 Dis_Buf[10] = {"        "}; //数码管显示缓存
  21. u8 Dis_Mode = 'C';  //数码管显示模式,‘C‘距离显示,’H’回显,‘F’盲区显示
  22. u8 hx_num = 0;//当前回显得是第几个
  23. u16 mq_num = 0;//盲区数值

  24. bit job1_flag = 0;//是否调用job1函数标志,在T1中断里面置1
  25. bit job2_flag = 0;//是否调用job2函数标志,在T1中断里面置1
  26. bit job3_flag = 0;//是否调用job3函数标志,在T1中断里面置1
  27. bit job4_flag = 0;//是否调用job4函数标志,在T1中断里面置1
  28. bit job5_flag = 0;//是否调用job5函数标志,在T1中断里面置1
  29. bit ssbz = 0;//闪烁标志

  30. sbit TX = P1^0;  //发射引脚
  31. sbit RX = P1^1;  //接收引脚

  32. void job1(void); //数码管动态显示处理,每5ms一次
  33. void job2(void); //显示模式处理,每100ms一次
  34. void job3(void); //按键读取与处理,每50ms一次
  35. void job4(void); //亮灯处理
  36. void job5(void); //电压输出
  37. void juli_set(void);//距离存储
  38. void juli_read(void);//距离读取
  39. void mq_read(void);//盲区读取
  40. void mq_set(void);//盲区存储
  41. void jlcl();//距离测量
  42. void send_wave(void);//发送超声波
  43. void dac_shuchu();//电压输出函数

  44. void Sys_Init()//系统初始化
  45. {
  46.         Buzzy_Off;
  47.         Relay_Off;
  48.         Led_Set = 0xff;
  49. }
  50. void main()
  51. {
  52.         Sys_Init();//系统初始化
  53.         Timer0Init();//T0初始化,T0用来测量距离
  54.         Timer1Init();//T1初始化,T1中断用来产生调用函数标志
  55. //        diyi_juli_h.f_num=dier_juli_h.f_num=disan_juli_h.f_num=disi_juli_h.f_num=0;//调试用
  56. //        juli_set();//调试用
  57.         juli_read();//读取eeprom数据
  58.         mq_read();//阈值读取
  59.         juli = disi_juli_h.f_num;//设置这次距离为第4次测量的距离
  60.         juli_s = disan_juli_h.f_num;//设置上次距离为第3次测量的距离
  61.         init_pcf8591();//初始化
  62.         while (1)//循环判断
  63.         {
  64.                 if (job1_flag)//判断时候为1,等于则调用函数,并将调用函数标志清0(下同)
  65.                 {
  66.                         job1(); //数码管动态显示处理,每5ms一次
  67.                         job1_flag = 0;//清0
  68.                 }
  69.                 if (job2_flag)
  70.                 {
  71.                         job2(); //显示模式处理,每100ms一次
  72.                         job2_flag = 0;
  73.                 }
  74.                 if (job3_flag)
  75.                 {
  76.                         job3(); //按键读取与处理,每50ms一次
  77.                         job3_flag = 0;
  78.                 }
  79.                 if (job4_flag)
  80.                 {
  81.                         job4();//LED处理函数
  82.                         job4_flag = 0;
  83.                 }
  84.                 if (job5_flag)
  85.                 {
  86.                         job5();//电压输出处理
  87.                         job5_flag = 0;
  88.                 }               
  89.         }
  90. }

  91. void job1(void) //数码管动态显示处理,每5ms一次
  92. {
  93.         Display(Dis_Buf);
  94. }

  95. void job2(void) //显示模式处理,每100ms一次,Work_Mode=’U’回显,其他常规显示
  96. {
  97.         switch (Dis_Mode) //判断当前显示模式
  98.         {
  99.                 case 'C': sprintf(Dis_Buf, "C %03d%03d",juli,juli_s);break;
  100.                 case 'F' : sprintf(Dis_Buf, "F     %02d",mq_num);break;
  101.                 case 'H':
  102.                 {
  103.                         switch (hx_num)
  104.                         {
  105.                                 case 0:
  106.                                 {
  107.                                         sprintf(Dis_Buf, "1    %03d",diyi_juli_h.f_num);        
  108.                                 }break;
  109.                                 case 1:
  110.                                 {
  111.                                         sprintf(Dis_Buf, "2    %03d",dier_juli_h.f_num);        
  112.                                 }break;
  113.                                 case 2:
  114.                                 {
  115.                                         sprintf(Dis_Buf, "3    %03d",disan_juli_h.f_num);        
  116.                                 }break;
  117.                                 case 3:
  118.                                 {
  119.                                         sprintf(Dis_Buf, "4    %03d",disi_juli_h.f_num);        
  120.                                 }break;
  121.                         }
  122.                 }break;
  123.         }
  124. }

  125. void job3(void) //按键读取与处理,每50ms一次
  126. {
  127.         u8 Key_Num = read_keyboard();
  128.         switch (Key_Num)
  129.         {
  130.                 case '1'://测量距离,循环覆盖,距离储存
  131.                 {
  132.                         juli_s = juli;
  133.                         jlcl();
  134.                         diyi_juli_h.f_num = dier_juli_h.f_num;
  135.                         dier_juli_h.f_num = disan_juli_h.f_num;
  136.                         disan_juli_h.f_num        = juli_s;
  137.                         disi_juli_h.f_num = juli;
  138.                         juli_set();
  139.                         ssbz = 1;
  140.                 }break;
  141.                 case 'C':
  142.                 {
  143.                         Dis_Mode = 'C';
  144.                 }break;
  145.                 case 'H': //S5 存储 (温度/电压/频率 存储到E2PROM)
  146.                 {
  147.                         Dis_Mode = 'H';
  148.                         juli_read();
  149.                 }break;
  150.                 case 'F':
  151.                 {
  152.                         Dis_Mode = 'F';
  153.                 }break;        
  154.         }        
  155. }

  156. void job4(void) //LED亮灯处理,每200ms一次
  157. {
  158.         static u8 time_num = 0;//时间标志
  159.         static u8 cs_num = 0;//亮灭次数
  160.         static u8 Led_Set_Temp = 0xff;//LED要显示的值
  161.         if(ssbz)//闪烁标志判断
  162.         {
  163.                 if(cs_num == 0)Led_Set_Temp = 0XFE;        //亮灭次数判断,如果第一次进来,就赋值。
  164.                 time_num++;//时间,如果等于5说明1S到了,要对亮的灯进行取反
  165.                 if(time_num == 5)
  166.                 {
  167.                         time_num = 0;
  168.                         if(Led_Set_Temp == 0XFE)Led_Set_Temp = 0XFF;                                       
  169.                         else if(Led_Set_Temp == 0XFF)Led_Set_Temp = 0XFE;
  170.                         else if(Led_Set_Temp == 0XBF)Led_Set_Temp = 0XBE;
  171.                         else if(Led_Set_Temp == 0XBE)Led_Set_Temp = 0XBF;
  172.                         else if(Led_Set_Temp == 0X7F)Led_Set_Temp = 0X7E;
  173.                         else if(Led_Set_Temp == 0X7E)Led_Set_Temp = 0X7F;
  174.                         cs_num++;//亮灭次数+1
  175.                         if(cs_num == 5)//如果等于5,说明已经亮灭循环3次,要将闪烁标志清零,亮灭次数清零
  176.                         {
  177.                                 cs_num = 0;
  178.                                 ssbz = 0;
  179.                         }
  180.                 }
  181.                 if(Dis_Mode == 'F')Led_Set_Temp &= 0XBF;
  182.                 else if(Dis_Mode == 'H')Led_Set_Temp &= 0X7F;        
  183.         }
  184.         else if(Dis_Mode == 'F')Led_Set_Temp = 0XBF;        
  185.         else if(Dis_Mode == 'H')Led_Set_Temp = 0X7F;
  186.         else Led_Set_Temp = 0xFF;
  187.         Led_Set=Led_Set_Temp;
  188. }

  189. void job5(void) //电压输出处理
  190. {
  191.         unsigned char dy = 0X00;
  192.         if(juli <= mq_num)
  193.         {
  194.                 dac_pcf8591(0X00);
  195.         }
  196.         else if(juli>mq_num)
  197.         {
  198.                 if((juli-mq_num)>=250)
  199.                         dy=255;
  200.                 else
  201.                         dy = (unsigned char)(((juli-mq_num)*0.02)/5.0*255.0);
  202.         }
  203.         dac_pcf8591(dy);               
  204. }


  205. void juli_set(void)//历史测量的距离储存
  206. {        
  207.         write_eeprom(0,diyi_juli_h.u_num[0]);i2c_delay(200);i2c_delay(200);
  208.         write_eeprom(1,diyi_juli_h.u_num[1]);i2c_delay(200);i2c_delay(200);

  209.         write_eeprom(2,dier_juli_h.u_num[0]);i2c_delay(200);i2c_delay(200);
  210.         write_eeprom(3,dier_juli_h.u_num[1]);i2c_delay(200);i2c_delay(200);

  211.         write_eeprom(4,disan_juli_h.u_num[0]);i2c_delay(200);i2c_delay(200);
  212.         write_eeprom(5,disan_juli_h.u_num[1]);i2c_delay(200);i2c_delay(200);

  213.         write_eeprom(6,disi_juli_h.u_num[0]);i2c_delay(200);i2c_delay(200);
  214.         write_eeprom(7,disi_juli_h.u_num[1]);i2c_delay(200);i2c_delay(200);
  215. }
  216. void juli_read(void)//历史测量距离读取
  217. {        
  218.         diyi_juli_h.u_num[0]=read_eeprom(0);i2c_delay(200);i2c_delay(200);
  219.         diyi_juli_h.u_num[1]=read_eeprom(1);i2c_delay(200);i2c_delay(200);
  220.         
  221.         dier_juli_h.u_num[0]=read_eeprom(2);i2c_delay(200);i2c_delay(200);
  222.         dier_juli_h.u_num[1]=read_eeprom(3);i2c_delay(200);i2c_delay(200);
  223.         
  224.         disan_juli_h.u_num[0]=read_eeprom(4);i2c_delay(200);i2c_delay(200);
  225.         disan_juli_h.u_num[1]=read_eeprom(5);i2c_delay(200);i2c_delay(200);        

  226.         disi_juli_h.u_num[0]=read_eeprom(6);i2c_delay(200);i2c_delay(200);
  227.         disi_juli_h.u_num[1]=read_eeprom(7);i2c_delay(200);i2c_delay(200);        
  228. }
  229. void mq_read()//盲区距离读取
  230. {
  231.         mq_num = read_eeprom(8);i2c_delay(200);i2c_delay(200);
  232. }

  233. void mq_set()//盲区距离存储
  234. {
  235.         write_eeprom(8,mq_num);i2c_delay(200);i2c_delay(200);
  236. }

  237. void jlcl()//距离测量函数
  238. {
  239.         unsigned int t = 0;
  240.         TR1 = 0;//停止T1计时,防止中断函数,干扰测距
  241.         send_wave();  //发送方波信号
  242.     TR0 = 1;  //启动计时
  243.         while((RX == 1) && (TF0 == 0));//等待收到脉冲
  244.         TR0 = 0;  //关闭计时
  245.         //发生溢出
  246.         if(TF0 == 1)
  247.         {
  248.                 TF0 = 0;
  249.         }
  250.         else
  251.         {
  252.                 /**  计算时间  */
  253.                 t = TH0;
  254.                 t <<= 8;
  255.                 t |= TL0;
  256.                 juli = (unsigned int)(t*0.017);  //计算距离                                
  257.         }
  258.         TH0 = 0;
  259.         TL0 = 0;
  260.         //juli += 10;
  261.         TR1 = 1;//启动T1计时
  262. }

  263. void send_wave(void)//发送脉冲函数
  264. {
  265.         unsigned char i = 8;  //发送8个脉冲
  266.         do
  267.         {
  268.                 TX = 1;
  269.                 somenop;somenop;somenop;somenop;somenop;somenop;
  270.                 somenop;somenop;somenop;somenop;               
  271.                 TX = 0;
  272.                 somenop;somenop;somenop;somenop;somenop;somenop;
  273.                 somenop;somenop;somenop;somenop;               
  274.         }
  275.         while(i--);
  276. }
复制代码
我就不把全部的程序贴上了,大家请到附件里面下载,还有就是,参加蓝桥杯比赛的同学可以关注我,我会在蓝桥杯国赛之前把所有的国赛题目和程序发上来(不参加的也可以关注哦 第8届:超声波测距机.rar (963.25 KB, 下载次数: 80)

评分

参与人数 1黑币 +70 收起 理由
admin + 70 赞一个!

查看全部评分

回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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