找回密码
 立即注册

QQ登录

只需一步,快速开始

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

超声测距程序一点反应也没有,有请高手

[复制链接]
跳转到指定楼层
楼主
ID:222521 发表于 2017-7-26 19:23 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
我在网上找的超声测距程序,实验板是我自己做的89c52RC+74HC595两片驱动6位共阳数码管。板子可以跑计数程序,IO口也没问题但就是上这个程序一点反应也没有,有请高人帮我看看是这程序有问题,还是我淘宝的HC-SR04有问题、
  1. #include <AT89X51.H>                                                                //头文件

  2. unsigned char code fseg[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};
  3. unsigned char code segbit[]={0x80,0x40,0x20,0x10,0x08,0x04,0x02,0x01};
  4. unsigned char  disbuf[8]={0,0,0,0,0,0,0,0};


  5. //-----------------------------------------------------------------------------
  6. // 函数原形定义
  7. #define uchar unsigned char
  8. #define uint unsigned int

  9. void main (void);                    // 主函数
  10. void LED4_Display (void);            // LED显示
  11. void LED_OUT(uchar X);                // LED单字节串行移位函数
  12. void delayms(uint);                    //延时子函数 ms
  13. void jisuan(void);

  14. unsigned char code LED_0F[];        // LED字模表

  15. sbit DIO=P1^0;  //14脚              //串行数据输入
  16. sbit RCLK=P1^1;//12脚                //时钟脉冲信号——上升沿有效
  17. sbit SCLK=P1^2;//11脚                //打入信号————上升沿有效
  18. sbit echo=P3^2;                                                    //echo
  19. sbit trig=P3^3;                                                    //trig



  20. //-----------------------------------------------------------------------------
  21. // 全局变量
  22. uchar LED[8];    //用于LED的8位显示缓存
  23. uint temp;                                                                                       
  24. uint temp1;
  25. uint a,b;                                                            //定义一个变量a,b  后者用于判断是否收到信号
  26. //
  27. // 主程序
  28. //
  29. void main(void)                                        //主函数开始
  30. {   
  31.         uint f;
  32.     echo=0;                                            //先拉低echo,trig引脚
  33.     trig=0;
  34.         f=500;
  35.     while(f>0);                                        //启动延时  消除第一次上电产生的波动
  36.         {
  37.            LED[3]=16;
  38.       LED[2]=15;
  39.       LED[1]=15;
  40.       LED[0]=16;
  41.          f--;
  42.        }
  43.     EA=1;                                            //开总中断
  44.     TMOD=0x11;                                        //设置定时器为方式1
  45.     ET0=1;        //允许定时器中断  这里主要是防止超声波模块未发送信号
  46.     ET1=1;
  47.     while(1)
  48.     {
  49.         echo=0;//a赋值
  50.         a=0;        
  51.         b=1;
  52.         TH0=0;                                        //定时器装初值
  53.         TL0=0;
  54.         TH1=(65536-25000)/256;                        //定时器装初值
  55.         TL1=(65536-25000)%256;
  56.         trig=1;                                        //trig送高
  57.         LED4_Display ();                                //    延时3ms
  58.         LED4_Display ();
  59.         LED4_Display ();
  60.         LED4_Display ();
  61.         LED4_Display ();
  62.         LED4_Display ();
  63.         LED4_Display ();
  64.         trig=0;                                                                                   //trig送低
  65.         TR1=1;
  66.         while(echo==0);                                    //等待echo变为高电平
  67.                 if(b==1)                                                                          // 判断是否收到信号
  68.         {

  69.             TR1=0;                                        //关定时器1
  70.             EX0=1;                                        //开外部中断
  71.             TR0=1;                                        //启动定时器
  72.             while(a==0);//注意这里!    前面给a赋0 程序停在这里等待中断
  73. //本来是在这里加上扫描屏幕程序的  发现有重大BUG索性删除  
  74.         }
  75.         else
  76.         {

  77.             LED[3]=16;
  78.             LED[2]=15;
  79.             LED[1]=15;
  80.             LED[0]=16;
  81.         LED4_Display ();
  82.         LED4_Display ();
  83.         LED4_Display ();
  84.         LED4_Display ();
  85.         LED4_Display ();
  86.         LED4_Display ();


  87.         }
  88.     }
  89. }
  90. void waibu() interrupt 0                                //外部中断服务子程序
  91. {
  92.     temp=TH0;                                    //取出定时器的值
  93.     temp1=TL0;                                                                                    
  94.     EX0=0;                                        //关闭外部中断
  95.     TR0=0;                                        //关闭定时器
  96.     jisuan();                                        //运行计算子程序
  97.     a=1;//a赋值1    程序回到刚才的  while(a)  中因为a的值已变为1,程序从头开始
  98. }
  99. void time1() interrupt 1                                  //定时器中断服务子程序
  100. {
  101.     TH0=0;                                        //重装初值
  102.     TL0=0;
  103. }

  104. void time2() interrupt 3
  105. {
  106.     TR1=0;
  107.     TH0=(65536-25000)/256;                        //定时器装初值
  108.     TL0=(65536-25000)%256;
  109.     b=0;
  110.     echo=1;

  111. }
  112. void jisuan(void)//计算子程序
  113. {
  114.     uint c,d;//定义一个变量c,d 用来判断距离
  115.     c=0;  //赋值0
  116.     d=0;   //给b重新赋值
  117.     LED4_Display (); //扫描一下数码管
  118.     temp=(temp<<8)+temp1;                            //TH0 TL0合并
  119.     temp=temp/5;//我没有精确计算 直接除5得出大概值

  120.     if(temp>40) //判段距离是否过近
  121.         {
  122.             c=1;
  123.         }
  124.         LED4_Display ();                            //扫描一下数码管
  125.     if(temp<40)
  126.         {
  127.          c=0;
  128.         }
  129.         LED4_Display ();
  130.         if(temp<3000) //判断距离是否过远
  131.         {
  132.             d=1;
  133.         }
  134.         LED4_Display ();
  135.     if(temp>3000)
  136.         {
  137.          d=0;
  138.         }
  139.     c=c&d;//与运算
  140.     if(c==1) //判断距离是否正常
  141.         {

  142.             LED[3]=temp/1000; //数值分离显示
  143.             LED[2]=temp%1000/100;
  144.             LED[1]=(temp%1000%100/10)+20;    //这个为什么要加上20呢?  因为这是个位  需要显示小数点  
  145.             LED[0]=temp%1000%100%10;
  146.         }
  147.     if(c==0)                                            //判断距离是否正常
  148.         {

  149.             LED[3]=16;
  150.             LED[2]=14;
  151.             LED[1]=14;
  152.             LED[0]=16;
  153.         }

  154. }
  155. void delayms(uint xms)                    //延时子函数 ms
  156.     {
  157.         uint i,j;
  158.         for(i=xms;i>0;i--)
  159.             for(j=110;j>0;j--);
  160.     }

  161.     //下面的程序是hc595模块显示程序
  162.     //每个模块的程序可以到资料的例程中移植
  163.     //*******************************************************//
  164. void LED4_Display (void)//屏幕扫描子函数   

  165. {
  166.     unsigned char code *led_table;          // 查表指针
  167.     uchar i;
  168.     //显示第1位
  169.     led_table = LED_0F + LED[0];
  170.     i = *led_table;

  171.     LED_OUT(i);            
  172.     LED_OUT(0xfe);        

  173.     RCLK = 0;
  174.     RCLK = 1;
  175.     //显示第2位
  176.     led_table = LED_0F + LED[1];
  177.     i = *led_table;

  178.     LED_OUT(i);        
  179.     LED_OUT(0x02);        

  180.     RCLK = 0;
  181.     RCLK = 1;
  182.     //显示第3位
  183.     led_table = LED_0F + LED[2];
  184.     i = *led_table;

  185.     LED_OUT(i);            
  186.     LED_OUT(0x04);   

  187.     RCLK = 0;
  188.     RCLK = 1;
  189.     //显示第4位
  190.     led_table = LED_0F + LED[3];
  191.     i = *led_table;

  192.     LED_OUT(i);            
  193.     LED_OUT(0x08);        

  194.     RCLK = 0;
  195.     RCLK = 1;

  196. }

  197. void LED_OUT(uchar X)
  198. {
  199.     uchar i;
  200.     for(i=8;i>=1;i--)
  201.     {
  202.         if (X&0x80) DIO=1; else DIO=0;
  203.         X<<=1;
  204.         SCLK = 0;
  205.         SCLK = 1;
  206.     }
  207. }
  208. //下面是显示数组
  209. unsigned char code LED_0F[] =
  210. {// 0      1       2       3     4      5       6       7     8      9       A       b     C    d      E    F     -        .
  211.     0xc0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0x8C,0xBF,0xC6,0xA1,0x86,0x8e,0xbf,0x7f,0x00,0x00,
  212. //    20    21     22   23   24   25   26    27   28  29   30
  213.     0x40,0x79,0x24,0x30,0x19,0x12,0x12,0x78,0x00,0x10,0xbf,
  214. };


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

使用道具 举报

沙发
ID:164602 发表于 2017-7-27 08:21 | 只看该作者
程序不对吧,特别是发射超声波,只需要至少10us,你用了18ms,天,回波早就过了,当然测不到了。我这里有一个例子,看看吧。
#include"reg51.h"
#include <intrins.h>//_nop_()函数头文件

sbit RX=P1^1;//连接Echo
sbit TX=P1^0;//连接Trig,

#define GPIO_DIG P0//数码管显示及控制I/O口定义
sbit LSA=P2^2;//138译码器的三位控制8位数码管
sbit LSB=P2^3;
sbit LSC=P2^4;

unsigned int  time=0;
unsigned int  timer=0;
unsigned long S=0;
bit flag=0;

unsigned char code DIG_CODE[16]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
unsigned char disbuff[4]={0,0,0,0};

void DigDisplay()//数码管显示函数,并赋初值0
{
        unsigned char i;
        unsigned int j;
        for(i=0;i<4;i++)
        {
                switch(i)
                {
                        case(0):
                                LSC=0;LSB=0;LSA=0; break;//显示第0位,最右位
                        case(1):
                                LSC=0;LSB=0;LSA=1; break;//显示第1位
                        case(2):
                                LSC=0;LSB=1;LSA=0; break;//显示第2位
                        case(3):
                                LSC=0;LSB=1;LSA=1; break;//显示第3位
                }
                GPIO_DIG=disbuff[i];//发送段码
                j=20;
                while(j--);       
                GPIO_DIG=0x00;
        }
}

void Conut(void)//计算距离函数
{
        time=TH0*256+TL0;
        TH0=0;//重装定时器0的初值
        TL0=0;
       
        S=(long)(time*0.17);
        if((S>=4000)||flag==1)
        {         
                flag=0;
                disbuff[0]=0x5c;//“o”
                disbuff[1]=0x72;//“r”
                disbuff[2]=0x72;//“r”
                disbuff[3]=0x79;//“E”
        }
        else
        {
                disbuff[3]=DIG_CODE[S%10000/1000];
                disbuff[2]=DIG_CODE[S%1000/100];
                disbuff[1]=DIG_CODE[S%100/10];
                disbuff[0]=DIG_CODE[S%10/1];
        }
}

void zd1() interrupt 1
{
        flag=1;
}

void zd3() interrupt 3
{
        TH1=0xf8;//重装初值
        TL1=0x30;
        DigDisplay();//调用显示
        timer++;
        if(timer>=50)
        {
                timer=0;
                TX=1;
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                TX=0;
        }
}

void  main(  void  )
{  
        TMOD=0x11;
        TH0=0;
        TL0=0;         
        TH1=0xf8;
        TL1=0x30;
        ET0=1;
        ET1=1;
        TR1=1;
        EA=1;

        while(1)
        {
                while(!RX);
                TR0=1;
                while(RX);
                TR0=0;
                Conut();
        }
}

可能与你的硬件显示部分不同,自己改改。

回复

使用道具 举报

板凳
ID:222521 发表于 2017-7-27 10:04 | 只看该作者
HC6800-ES-V2.0 发表于 2017-7-27 08:21
程序不对吧,特别是发射超声波,只需要至少10us,你用了18ms,天,回波早就过了,当然测不到了。我这里有一 ...

首先要感谢你的回复虽然我还不明白该如何改动,但我会去仔细琢磨,谢谢你的程序。
回复

使用道具 举报

地板
ID:222521 发表于 2017-7-27 19:29 | 只看该作者
好消息!好消息!现在已经可以跑程序了,只是显示的字符不正常,等我继续弄明白后告诉大家,不过看来大家对这个兴趣不大
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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