找回密码
 立即注册

QQ登录

只需一步,快速开始

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

HC-SR04超声波模块51+PIC单片机例程

[复制链接]
跳转到指定楼层
楼主
51单片机超详细的HC-SR04超声波模块例程

单片机源程序如下:

  1. //晶振=8M
  2. //MCU=STC10F04XE
  3. //P0.0-P0.6共阳数码管引脚
  4. //Trig  = P1^0
  5. //Echo  = P3^2
  6. #include <reg52.h>     //包括一个52标准内核的头文件
  7. #define uchar unsigned char //定义一下方便使用
  8. #define uint  unsigned int
  9. #define ulong unsigned long
  10. //***********************************************
  11. sfr  CLK_DIV = 0x97; //为STC单片机定义,系统时钟分频
  12.                      //为STC单片机的IO口设置地址定义
  13. sfr   P0M1   = 0X93;
  14. sfr   P0M0   = 0X94;
  15. sfr   P1M1   = 0X91;
  16. sfr   P1M0   = 0X92;
  17. sfr        P2M1   = 0X95;
  18. sfr        P2M0   = 0X96;
  19. //***********************************************
  20. sbit Trig  = P1^0; //产生脉冲引脚
  21. sbit Echo  = P3^2; //回波引脚
  22. sbit test  = P1^1; //测试用引脚

  23. uchar code SEG7[10]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90};//数码管0-9
  24. uint distance[4];  //测距接收缓冲区
  25. uchar ge,shi,bai,temp,flag,outcomeH,outcomeL,i;  //自定义寄存器
  26. bit succeed_flag;  //测量成功标志
  27. //********函数声明
  28. void conversion(uint temp_data);
  29. void delay_20us();
  30. //void pai_xu();

  31. void main(void)   // 主程序
  32. {  uint distance_data,a,b;
  33.    uchar CONT_1;   
  34.    CLK_DIV=0X03; //系统时钟为1/8晶振(pdf-45页)
  35.      P0M1 = 0;   //将io口设置为推挽输出
  36.      P1M1 = 0;
  37.      P2M1 = 0;
  38.      P0M0 = 0XFF;
  39.      P1M0 = 0XFF;
  40.      P2M0 = 0XFF;
  41.    i=0;
  42.    flag=0;
  43.         test =0;
  44.         Trig=0;       //首先拉低脉冲输入引脚
  45.         TMOD=0x11;    //定时器0,定时器1,16位工作方式
  46.         TR0=1;             //启动定时器0
  47.    IT0=0;        //由高电平变低电平,触发外部中断
  48.         ET0=1;        //打开定时器0中断
  49. //ET1=1;        //打开定时器1中断
  50.         EX0=0;        //关闭外部中断
  51.         EA=1;         //打开总中断0       
  52.   
  53.        
  54. while(1)         //程序循环
  55.         {
  56.   EA=0;
  57.              Trig=1;
  58.         delay_20us();
  59.         Trig=0;         //产生一个20us的脉冲,在Trig引脚  
  60.         while(Echo==0); //等待Echo回波引脚变高电平
  61.              succeed_flag=0; //清测量成功标志
  62.              EX0=1;          //打开外部中断
  63.                    TH1=0;          //定时器1清零
  64.         TL1=0;          //定时器1清零
  65.              TF1=0;          //
  66.         TR1=1;          //启动定时器1
  67.    EA=1;

  68.       while(TH1 < 30);//等待测量的结果,周期65.535毫秒(可用中断实现)  
  69.                   TR1=0;          //关闭定时器1
  70.         EX0=0;          //关闭外部中断

  71.     if(succeed_flag==1)
  72.              {        
  73.                    distance_data=outcomeH;                //测量结果的高8位
  74.            distance_data<<=8;                   //放入16位的高8位
  75.                      distance_data=distance_data|outcomeL;//与低8位合并成为16位结果数据
  76.             distance_data*=12;                  //因为定时器默认为12分频
  77.            distance_data/=58;                   //微秒的单位除以58等于厘米
  78.          }                                      //为什么除以58等于厘米,  Y米=(X秒*344)/2
  79.                                                                // X秒=( 2*Y米)/344 ==》X秒=0.0058*Y米 ==》厘米=微秒/58
  80.     if(succeed_flag==0)
  81.                    {
  82.             distance_data=0;                    //没有回波则清零
  83.                            test = !test;                       //测试灯变化
  84.            }

  85.      ///       distance[i]=distance_data; //将测量结果的数据放入缓冲区
  86.      ///        i++;
  87.             ///         if(i==3)
  88.           ///             {
  89.           ///               distance_data=(distance[0]+distance[1]+distance[2]+distance[3])/4;
  90.      ///        pai_xu();
  91.      ///        distance_data=distance[1];

  92.       
  93.            a=distance_data;
  94.        if(b==a) CONT_1=0;
  95.        if(b!=a) CONT_1++;
  96.        if(CONT_1>=3)
  97.                    { CONT_1=0;
  98.                           b=a;
  99.                           conversion(b);
  100.                         }      
  101.           ///                 i=0;
  102.           ///                }             
  103.          }
  104. }
  105. //***************************************************************
  106. //外部中断0,用做判断回波电平
  107. INTO_()  interrupt 0   // 外部中断是0号
  108. {   
  109.      outcomeH =TH1;    //取出定时器的值
  110.      outcomeL =TL1;    //取出定时器的值
  111.      succeed_flag=1;   //至成功测量的标志
  112.      EX0=0;            //关闭外部中断
  113.   }
  114. //****************************************************************
  115. //定时器0中断,用做显示
  116. timer0() interrupt 1  // 定时器0中断是1号
  117.    {
  118.          TH0=0xfd; //写入定时器0初始值
  119.          TL0=0x77;                
  120.          switch(flag)   
  121.       {case 0x00:P0=ge; P2=0xfd;flag++;break;
  122.             case 0x01:P0=shi;P2=0xfe;flag++;break;
  123.             case 0x02:P0=bai;P2=0xfb;flag=0;break;
  124.       }
  125.    }
  126. //*****************************************************************
  127. /*
  128. //定时器1中断,用做超声波测距计时
  129. timer1() interrupt 3  // 定时器0中断是1号
  130.     {
  131. TH1=0;
  132. TL1=0;
  133.      }
  134. */
  135. //******************************************************************
  136. //显示数据转换程序
  137. void conversion(uint temp_data)  
  138. {  
  139.     uchar ge_data,shi_data,bai_data ;
  140.     bai_data=temp_data/100 ;
  141.     temp_data=temp_data%100;   //取余运算
  142.     shi_data=temp_data/10 ;
  143.     temp_data=temp_data%10;   //取余运算
  144.     ge_data=temp_data;

  145.     bai_data=SEG7[bai_data];
  146.     shi_data=SEG7[shi_data];
  147.     ge_data =SEG7[ge_data];

  148.     EA=0;
  149.     bai = bai_data;
  150.     shi = shi_data;
  151.     ge  = ge_data ;
  152.          EA=1;
  153. }
  154. //******************************************************************
  155. void delay_20us()
  156. {  uchar bt ;
  157.     for(bt=0;bt<100;bt++);
  158. }
  159. /*
  160. void pai_xu()
  161.   {  uint t;
  162.   if (distance[0]>distance[1])
  163.     {t=distance[0];distance[0]=distance[1];distance[1]=t;} /*交换值
  164.   if(distance[0]>distance[2])
  165.     {t=distance[2];distance[2]=distance[0];distance[0]=t;} /*交换值
  166.   if(distance[1]>distance[2])
  167.     {t=distance[1];distance[1]=distance[2];distance[2]=t;} /*交换值         
  168.     }
  169. */
复制代码

所有资料51hei提供下载:
超声波测距资料.zip (1.21 MB, 下载次数: 66)


评分

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

查看全部评分

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

使用道具 举报

沙发
ID:373610 发表于 2018-11-28 15:08 | 只看该作者
感谢分享
回复

使用道具 举报

板凳
ID:31119 发表于 2020-7-14 16:45 | 只看该作者
看看咋样PIC的刚了解
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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