找回密码
 立即注册

QQ登录

只需一步,快速开始

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

超声波测距单片机程序

[复制链接]
跳转到指定楼层
楼主
ID:446636 发表于 2019-7-30 09:07 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
#include <reg52.H>//器件配置文件
#include <intrins.h>
//传感器接口
sbit RX  = P3^2;
sbit TX  = P3^3;
//按键声明
sbit S1  = P1^4;
sbit S2  = P1^5;
sbit S3  = P1^6;
//蜂鸣器
sbit Feng= P2^0;

sbit W1=P1^0;
sbit W2=P1^1;
sbit W3=P1^2;
sbit W4=P1^3;
//变量声明
unsigned int  time=0;
unsigned int  timer=0;
unsigned char posit=0;
unsigned long S=0;
unsigned long BJS=50;//报警距离80CM
//模式 0正常模式 1调整
char Mode=0;
bit  flag=0;
bit flag_KEY=0;
unsigned char const discode[] ={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x40,0xff/*-*/};        //数码管显示码0123456789-和不显示
//unsigned char const positon[4]={0xfd,0xfb,0xf7,0xfe};        //位选
unsigned char disbuff[4]           ={0,0,0,0};                 //数组用于存放距离信息
unsigned char disbuff_BJ[4]        ={0,0,0,0};//报警信息
void Display();
//延时20ms(不精确)
void delay(void)
{
    unsigned char a,b,c;
    for(c=2;c>0;c--)
        for(b=38;b>0;b--)
            for(a=60;a>0;a--);
}

//按键扫描
void Key_()
{
        if(flag_KEY==0)
        {
                if(Mode!=0)
                {
                        //+
                        if(S1==0)
                        {
                                delay();           //延时去抖
                                if(S1==0)
                                {
                                        BJS++;         //报警值加
                                        flag_KEY=1;
                                        if(BJS>=151) //最大151
                                        {
                                                BJS=0;
                                        }
//                                        while(S1==0)
//                                        Display();
                                }
                                
                        }
                        //-
                        if(S2==0)
                        {
                                delay();
                                if(S2==0)
                                {
                                        BJS--;         //报警值减
                                        flag_KEY=1;
                                        if(BJS<=1)         //最小1
                                        {
                                                BJS=150;
                                        }        
//                                        while(S2==0)
//                                        Display();
                                }
                                
                        }
                }
                //功能
                if(S3==0)                //设置键
                {
                        delay();
                        if(S3==0)
                        {
                                Mode++;                //模式加
                                flag_KEY=1;
                                if(Mode>=2)                //加到2时清零
                                {
                                        Mode=0;
                                }
//                                while(S3==0)
//                                Display();
                        }
                }
        }
        if((P1&0x70)==0x70)
        {
                flag_KEY=0;
        }
}
/**********************************************************************************************************/
//扫描数码管
void Display(void)                                 
{
        //正常显示
        if(Mode==0)
        {
                P0=0x00;  //关闭显示
                if(posit==1)//数码管的小数点
                {
                        P0=(discode[disbuff[posit]])|0x80;//按位或,最高位变为1,显示小数点
                }
                else if(posit==0)
                {
                        P0=~discode[11];
                }
                else
                {
                        P0=discode[disbuff[posit]];
                }
                switch(posit)
                {
                        case 0 : W1=0;W2=1;W3=1;W4=1; break;
                        case 1 : W1=1;W2=0;W3=1;W4=1; break;
                        case 2 : W1=1;W2=1;W3=0;W4=1; break;
                        case 3 : W1=1;W2=1;W3=1;W4=0; break;
                }
                posit++;
                if(posit>3)                //每进一次显示函数,变量加1
                        posit=0;                //加到3时清零
        }
        //报警显示
        else
        {
                P0=0x00;
                if(posit==1)//数码管的小数点
                {
                        P0=(discode[disbuff_BJ[posit]])|0x80;
                }
                else if(posit==0)
                {
                        P0=0x76;        //显示字母               
                }
                else
                {
                        P0=discode[disbuff_BJ[posit]];
                }
                switch(posit)
                {
                        case 0 : W1=0;W2=1;W3=1;W4=1; break;
                        case 1 : W1=1;W2=0;W3=1;W4=1; break;
                        case 2 : W1=1;W2=1;W3=0;W4=1; break;
                        case 3 : W1=1;W2=1;W3=1;W4=0; break;
                }
                posit++;
                if(posit>3)
                        posit=0;
        }
}
/**********************************************************************************************************/
//计算
void Conut(void)
{
        time=TH0*256+TL0;          //读出T0的计时数值
        TH0=0;
        TL0=0;                                  //清空计时器
        S=(time*1.7)/100;     //算出来是CM
        //声音的速度是340m/s,时间的单位是us,计算到秒需要将时间数据/1000000,
        //长度=速度*时间,340*time/1000000,长度数据单位是m转换成cm需要乘以100得到340*time/10000,
        //小数点都向左移两位得到3.4*time/100,因为超声波是往返了,所以再除以2,得到距离数据(time*1.7)/100
        if(Mode==0)                          //非设置状态时
        {
                if((S>=700)||flag==1) //超出测量范围显示“-”
                {        
                        Feng=0;                    //蜂鸣器报警
                        flag=0;
                        disbuff[1]=10;           //“-”
                        disbuff[2]=10;           //“-”
                        disbuff[3]=10;           //“-”
                }
                else
                {
                        //距离小于报警距
                        if(S<=BJS)
                        {
                                Feng=0;        //报警
                        }
                        else  //大于
                        {
                                Feng=1;                //关闭报警        
                        }
                        disbuff[1]=S%1000/100;                 //将距离数据拆成单个位赋值
                        disbuff[2]=S%1000%100/10;
                        disbuff[3]=S%1000%10 %10;
                }
        }
        else
        {
                        Feng=1;
                        disbuff_BJ[1]=BJS%1000/100;
                        disbuff_BJ[2]=BJS%1000%100/10;
                        disbuff_BJ[3]=BJS%1000%10 %10;
        }
}
/**********************************************************************************************************/
//定时器0
void zd0() interrupt 1                  //T0中断用来计数器溢出,超过测距范围
{
        flag=1;                                                 //中断溢出标志
}
/**********************************************************************************************************/
//定时器1
void zd3() interrupt 3                  //T1中断用来扫描数码管和计800MS启动模块
{
        TH1=0xf8;
        TL1=0x30;                                 //定时2ms
        Key_();                                        //扫描按键
        Display();                                //扫描显示
        timer++;                                //变量加
        if(timer>=400)                        //400次就是800ms
        {
                timer=0;
                TX=1;                                  //800MS  启动一次模块
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                TX=0;
        }
}
/**********************************************************************************************************/
//主函数
void main(void)
{  
        TMOD=0x11;                   //设T0为方式1
        TH0=0;
        TL0=0;         
        TH1=0xf8;                   //2MS定时
        TL1=0x30;
        ET0=1;                                //允许T0中断
        ET1=1;                           //允许T1中断
        TR1=1;                           //开启定时器
        EA=1;                                        //开启总中断        
        while(1)
        {
                while(!RX);                //当上次接收完波后,RX引脚是低电平,取反就是1,此while成立,反复判断RX状态。当RX没有接收到返回波时是高电平,取反就是0,此while不成立,跳出
                TR0=1;                        //开启计数
                while(RX);                //当RX没有接收到返回波,此while成立,程序停在这里一直判断RX状态。当RX接收到返回波,RX引脚变为低电平,此while不成立,跳出
                TR0=0;                        //停止计数
                Conut();                        //计算
        }
}

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

使用道具 举报

沙发
ID:592870 发表于 2019-7-31 10:00 | 只看该作者
楼主,能否提供一下仿真电路图。
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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