找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2698|回复: 1
打印 上一主题 下一主题
收起左侧

基于51单片机超声波雷达测距(源程序+Proteus仿真)

[复制链接]
跳转到指定楼层
楼主
超声波雷达测距
程序+Proteus仿真工程


单片机源程序如下:
  1. #include <reg52.H>//器件配置文件
  2. #include <intrins.h>
  3. #include "eeprom52.h"
  4. //传感器接口
  5. sbit RX  = P3^2;
  6. sbit TX  = P3^3;
  7. //按键声明
  8. sbit S1  = P1^4;
  9. sbit S2  = P1^5;
  10. sbit S3  = P1^6;
  11. //蜂鸣器
  12. sbit Feng= P2^0;

  13. sbit W1=P1^0;
  14. sbit W2=P1^1;
  15. sbit W3=P1^2;
  16. sbit W4=P1^3;
  17. //变量声明
  18. unsigned int  time=0;
  19. unsigned int  timer=0;
  20. unsigned char posit=0;
  21. unsigned long S=0;
  22. unsigned long BJS;//报警距离
  23. //模式 0正常模式 1调整
  24. char Mode=0;
  25. bit  flag=0;
  26. bit flag_KEY=0;
  27. unsigned char const discode[] ={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x40,0xff/*-*/};        //数码管显示码0123456789-和不显示
  28. //unsigned char const positon[4]={0xfd,0xfb,0xf7,0xfe};        //位选
  29. unsigned char disbuff[4]           ={0,0,0,0};                 //数组用于存放距离信息
  30. unsigned char disbuff_BJ[4]        ={0,0,0,0};//报警信息
  31. void Display();

  32. /******************把数据保存到单片机内部eeprom中******************/
  33. void write_eeprom()
  34. {
  35.         SectorErase(0x2000);
  36.         byte_write(0x2000, BJS);
  37.         byte_write(0x2060, a_a);        
  38. }

  39. /******************把数据从单片机内部eeprom中读出来*****************/
  40. void read_eeprom()
  41. {
  42.         BJS   = byte_read(0x2000);
  43.         a_a      = byte_read(0x2060);
  44. }

  45. /**************开机自检eeprom初始化*****************/
  46. void init_eeprom()
  47. {
  48.         read_eeprom();                //先读
  49.         if(a_a != 1)                //新的单片机初始单片机内问eeprom
  50.         {
  51.                 BJS   = 50;
  52.                 a_a = 1;
  53.                 write_eeprom();           //保存数据
  54.         }        
  55. }

  56. //延时20ms(不精确)
  57. void delay(void)
  58. {
  59.     unsigned char a,b,c;
  60.     for(c=2;c>0;c--)
  61.         for(b=38;b>0;b--)
  62.             for(a=60;a>0;a--);
  63. }

  64. //按键扫描
  65. void Key_()
  66. {
  67.         if(flag_KEY==0)
  68.         {
  69.                 if(Mode!=0)
  70.                 {
  71.                         //+
  72.                         if(S1==0)
  73.                         {
  74.                                 delay();           //延时去抖
  75.                                 if(S1==0)
  76.                                 {
  77.                                         BJS++;         //报警值加
  78.                                         flag_KEY=1;
  79.                                         if(BJS>=151) //最大151
  80.                                         {
  81.                                                 BJS=0;
  82.                                         }
  83.                                         write_eeprom();//保存数据
  84. //                                        while(S1==0)
  85. //                                        Display();
  86.                                 }
  87.                                 
  88.                         }
  89.                         //-
  90.                         if(S2==0)
  91.                         {
  92.                                 delay();
  93.                                 if(S2==0)
  94.                                 {
  95.                                         BJS--;         //报警值减
  96.                                         flag_KEY=1;
  97.                                         if(BJS<=1)         //最小1
  98.                                         {
  99.                                                 BJS=150;
  100.                                         }
  101.                                         write_eeprom();//保存数据        
  102. //                                        while(S2==0)
  103. //                                        Display();
  104.                                 }
  105.                                 
  106.                         }
  107.                 }
  108.                 //功能
  109.                 if(S3==0)                //设置键
  110.                 {
  111.                         delay();
  112.                         if(S3==0)
  113.                         {
  114.                                 Mode++;                //模式加
  115.                                 flag_KEY=1;
  116.                                 if(Mode>=2)                //加到2时清零
  117.                                 {
  118.                                         Mode=0;
  119.                                 }
  120. //                                while(S3==0)
  121. //                                Display();
  122.                         }
  123.                 }
  124.         }
  125.         if((P1&0x70)==0x70)
  126.         {
  127.                 flag_KEY=0;
  128.         }
  129. }
  130. /**********************************************************************************************************/
  131. //扫描数码管
  132. void Display(void)                                 
  133. {
  134.         //正常显示
  135.         if(Mode==0)
  136.         {
  137.                 P0=0x00;  //关闭显示
  138.                 if(posit==1)//数码管的小数点
  139.                 {
  140.                         P0=(discode[disbuff[posit]])|0x80;//按位或,最高位变为1,显示小数点
  141.                 }
  142.                 else if(posit==0)
  143.                 {
  144.                         P0=~discode[11];
  145.                 }
  146.                 else
  147.                 {
  148.                         P0=discode[disbuff[posit]];
  149.                 }
  150.                 switch(posit)
  151.                 {
  152.                         case 0 : W1=0;W2=1;W3=1;W4=1; break;
  153.                         case 1 : W1=1;W2=0;W3=1;W4=1; break;
  154.                         case 2 : W1=1;W2=1;W3=0;W4=1; break;
  155.                         case 3 : W1=1;W2=1;W3=1;W4=0; break;
  156.                 }
  157.                 posit++;
  158.                 if(posit>3)                //每进一次显示函数,变量加1
  159.                         posit=0;                //加到3时清零
  160.         }
  161.         //报警显示
  162.         else
  163.         {
  164.                 P0=0x00;
  165.                 if(posit==1)//数码管的小数点
  166.                 {
  167.                         P0=(discode[disbuff_BJ[posit]])|0x80;
  168.                 }
  169.                 else if(posit==0)
  170.                 {
  171.                         P0=0x76;        //显示字母               
  172.                 }
  173.                 else
  174.                 {
  175.                         P0=discode[disbuff_BJ[posit]];
  176.                 }
  177.                 switch(posit)
  178.                 {
  179.                         case 0 : W1=0;W2=1;W3=1;W4=1; break;
  180.                         case 1 : W1=1;W2=0;W3=1;W4=1; break;
  181.                         case 2 : W1=1;W2=1;W3=0;W4=1; break;
  182.                         case 3 : W1=1;W2=1;W3=1;W4=0; break;
  183.                 }
  184.                 posit++;
  185.                 if(posit>3)
  186.                         posit=0;
  187.         }
  188. }
  189. /**********************************************************************************************************/
  190. //计算
  191. void Conut(void)
  192. {
  193.         time=TH0*256+TL0;          //读出T0的计时数值
  194.         TH0=0;
  195.         TL0=0;                                  //清空计时器
  196.         S=(time*1.7)/100;     //算出来是CM
  197.         //声音的速度是340m/s,时间的单位是us,计算到秒需要将时间数据/1000000,
  198.         //长度=速度*时间,340*time/1000000,长度数据单位是m转换成cm需要乘以100得到340*time/10000,
  199.         //小数点都向左移两位得到3.4*time/100,因为超声波是往返了,所以再除以2,得到距离数据(time*1.7)/100
  200.         if(Mode==0)                          //非设置状态时
  201.         {
  202.                 if((S>=700)||flag==1) //超出测量范围显示“-”
  203.                 {        
  204.                         Feng=0;                    //蜂鸣器报警
  205.                         flag=0;
  206.                         disbuff[1]=10;           //“-”
  207.                         disbuff[2]=10;           //“-”
  208.                         disbuff[3]=10;           //“-”
  209.                 }
  210.                 else
  211.                 {
  212.                         //距离小于报警距
  213.                         if(S<=BJS)
  214.                         {
  215.                                 Feng=0;        //报警
  216.                         }
  217.                         else  //大于
  218.                         {
  219.                                 Feng=1;                //关闭报警        
  220.                         }
  221.                         disbuff[1]=S%1000/100;                 //将距离数据拆成单个位赋值
  222.                         disbuff[2]=S%1000%100/10;
  223.                         disbuff[3]=S%1000%10 %10;
  224.                 }
  225.         }
  226.         else
  227.         {
  228.                         Feng=1;
  229.                         disbuff_BJ[1]=BJS%1000/100;
  230.                         disbuff_BJ[2]=BJS%1000%100/10;
  231.                         disbuff_BJ[3]=BJS%1000%10 %10;
  232.         }
  233. }
  234. /**********************************************************************************************************/
  235. //定时器0
  236. void zd0() interrupt 1                  //T0中断用来计数器溢出,超过测距范围
  237. {
  238.         flag=1;                                                 //中断溢出标志
  239. }
  240. /**********************************************************************************************************/
  241. //定时器1
  242. void zd3() interrupt 3                  //T1中断用来扫描数码管和计800MS启动模块
  243. {
  244.         TH1=0xf8;
  245.         TL1=0x30;                                 //定时2ms
  246.         Key_();                                        //扫描按键
  247.         Display();                                //扫描显示
  248.         timer++;                                //变量加
  249.         if(timer>=400)                        //400次就是800ms
  250.         {
  251.                 timer=0;
  252.                 TX=1;                                  //800MS  启动一次模块
  253.                 _nop_();
  254.                 _nop_();
  255.                 _nop_();
  256.                 _nop_();
  257.                 _nop_();
  258.                 _nop_();
  259.                 _nop_();
  260.                 _nop_();
  261.                 _nop_();
  262.                 _nop_();
  263.                 _nop_();
  264.                 _nop_();
  265.                 _nop_();
  266.                 _nop_();
  267.                 _nop_();
  268.                 _nop_();
  269.                 _nop_();
  270.                 _nop_();
  271.                 _nop_();
  272.                 _nop_();
  273.                 _nop_();
  274.                 TX=0;
  275.         }
  276. }
  277. /**********************************************************************************************************/
  278. //主函数
  279. void main(void)
  280. {  
  281.         TMOD=0x11;                   //设T0为方式1
  282.         TH0=0;
  283.         TL0=0;         
  284.         TH1=0xf8;                   //2MS定时
  285.         TL1=0x30;
  286.         ET0=1;                                //允许T0中断
  287.         ET1=1;                           //允许T1中断
  288.         TR1=1;                           //开启定时器
  289.         EA=1;                                        //开启总中断        
  290.         init_eeprom();  //开始初始化保存的数据
  291.         while(1)
  292.         {
  293.                 while(!RX);                //当上次接收完波后,RX引脚是低电平,取反就是1,此while成立,反复判断RX状态。当RX没有接收到返回波时是高电平,取反就是0,此while不成立,跳出
  294.                 TR0=1;                        //开启计数
  295.                 while(RX);                //当RX没有接收到返回波,此while成立,程序停在这里一直判断RX状态。当RX接收到返回波,RX引脚变为低电平,此while不成立,跳出
  296.                 TR0=0;                        //停止计数
  297.                 Conut();                        //计算
  298.         }
  299. }
复制代码

仿真程序资料51hei下载地址:
超声波测距.zip (160.74 KB, 下载次数: 54)


评分

参与人数 1黑币 +20 收起 理由
admin + 20 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:952332 发表于 2021-7-12 22:02 | 只看该作者
请问一下,什么是器配置文件啊?
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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